我们都知道boost和c++11都支持shared_ptr。有些编译器支持 c++11,有些不支持。我想编写我的代码,以便当编译器支持 c++11 shared_ptr 时,它使用 std::shared_ptr; 如果没有,请使用 boost::shared_ptr。什么是常见/最佳实践?
让我将讨论限制在 GCC 上,而不是特定版本。
我们都知道boost和c++11都支持shared_ptr。有些编译器支持 c++11,有些不支持。我想编写我的代码,以便当编译器支持 c++11 shared_ptr 时,它使用 std::shared_ptr; 如果没有,请使用 boost::shared_ptr。什么是常见/最佳实践?
让我将讨论限制在 GCC 上,而不是特定版本。
截至今天,我知道检测 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++11 支持在整个 GCC 版本中发生了很大变化,不仅在核心语言级别,而且在库级别也是如此.
我最好的选择是找到您认为可以接受 C++11 支持的最小 GCC 版本,并将上述测试与 GCC 版本测试结合起来,例如:
#if (((__GNUC__ == 4) && (__GNUC_MINOR__ >= 7)) || (__GNUC__ >= 5)) // for GCC 4.7+
boost
在和之间切换std
现在,在两者之间切换有点麻烦boost::shared_ptr
,std::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 的方式。
您可以将它们拉入自定义命名空间
#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++ 实现还提供了功能测试宏,以便快速轻松地检测新功能。