7

我们都知道boost和c++11都支持shared_ptr。有些编译器支持 c++11,有些不支持。我想编写我的代码,以便当编译器支持 c++11 shared_ptr 时,它使用 std::shared_ptr; 如果没有,请使用 boost::shared_ptr。什么是常见/最佳实践?

让我将讨论限制在 GCC 上,而不是特定版本。

4

2 回答 2

4

C++0x / C++11 可用性

截至今天,我知道检测 GCC 是否使用 C++0x/C++11 的唯一方法是检查预定义的宏__GXX_EXPERIMENTAL_CXX0X__

#ifdef __GXX_EXPERIMENTAL_CXX0X__
    // C++11 code
#else
    // C++03 code
#endif

请注意,这可能会在未来发生变化(因此EXPERIMENTAL是宏的一部分)。

编辑:事实上,在@stephan 在评论中指出之前,我不知道有一种更好的方法:

16.8/1 [cpp.predefined]

以下宏名称应由实现定义:

__cplusplus

该名称在编译 C++ 翻译单元时__cplusplus定义为值。157201103L

157) 本标准的未来版本打算用更大的值替换此宏的值。不符合标准的编译器应使用最多五个十进制数字的值。

所以我想你可以这样做:

#if defined(__cplusplus) && (__cplusplus >= 201103L)

但我担心这不适用于标准 C++0x GCC 版本。所以我可能会同时使用__GXX_EXPERIMENTAL_CXX0X__两者__cplusplus

#if defined(__GXX_EXPERIMENTAL_CXX0X__) || (defined(__cplusplus) && (__cplusplus >= 201103L))

C++0x / C++11 兼容性

然而,仅仅检测是否启用了 C++0x/C++11 支持可能并不总是足够:C++11 支持在整个 GCC 版本中发生了很大变化,不仅在核心语言级别,而且在库级别也是如此.

我最好的选择是找到您认为可以接受 C++11 支持的最小 GCC 版本,并将上述测试与 GCC 版本测试结合起来,例如:

#if (((__GNUC__ == 4) && (__GNUC_MINOR__ >= 7)) || (__GNUC__ >= 5)) // for GCC 4.7+

boost在和之间切换std

现在,在两者之间切换有点麻烦boost::shared_ptrstd::shared_ptr因为 C++03 不支持模板化类型定义。我会将所需的定义包装在 a 中template struct

#if (defined(__GXX_EXPERIMENTAL_CXX0X__) || (defined(__cplusplus) && (__cplusplus >= 201103L))) \
        && (((__GNUC__ == 4) && (__GNUC_MINOR__ >= 7)) || (__GNUC__ >= 5))
    template<typename T>
    struct my_shared_ptr {
        typedef std::shared_ptr<T> type;
        // you may also want to forward make_shared etc, this is left as an exercise
    };
#else
    template<typename T>
    struct my_shared_ptr {
        typedef boost::shared_ptr<T> type;
        // you may also want to forward make_shared etc, this is left as an exercise
    };
#endif

my_shared_ptr<int>::type ptr(new int);

编辑:(再次)哎呀,我刚刚注意到@ComicSansMSshared_ptr使用简单的非模板using指令导入权利的方式。不知道为什么我没有想到这一点。

尽管如此,我仍然支持我描述的在 GCC 上检测 C++0x/C++11 的方式。

于 2013-06-15T19:37:59.867 回答
3

您可以将它们拉入自定义命名空间

#ifdef HAS_STD_SHARED_PTR
    #include <memory>
    #define SHARED_PTR_NAMESPACE std
#else
    #include <boost/shared_ptr.hpp>
    #define SHARED_PTR_NAMESPACE boost
#endif

namespace my_namespace {
     using SHARED_PTR_NAMESPACE::shared_ptr;
     using SHARED_PTR_NAMESPACE::make_shared;
}

#undef SHARED_PTR_NAMESPACE

HAS_STD_SHARED_PTR标志需要由构建环境设置。CMake 提供了多种检测此类功能的机制,较新的 C++ 实现还提供了功能测试宏,以便快速轻松地检测新功能。

于 2013-06-15T19:33:32.297 回答