8

我想获得一种模板化的方式来查找类型是否为 shared_ptr 并且基于此我想要对函数进行新的专业化。

示例主要功能是,

template <class T> inline
void CEREAL_LOAD_FUNCTION_NAME( RelaxedJSONInputArchive & ar,    NameValuePair<T> & t )
{
    std::cout << " CEREAL_LOAD_FUNCTION_NAME NameValuePair 1 " << std::endl;
     ar.setNextName( t.name );
     ar( t.value );
}

如果 t.value 是 shared_ptr 那么我想要一个不同的函数专业化。我在下面尝试过,

template <class T> inline
typename std::enable_if<is_pointer<T>::value, void>::type
CEREAL_LOAD_FUNCTION_NAME( RelaxedJSONInputArchive & ar, NameValuePair<T> & t )
 {
    std::cout << " CEREAL_LOAD_FUNCTION_NAME NameValuePair 2 " << std::endl;
   ar.setNextName( t.name );
   ar( t.value );
  }

但这似乎不起作用。这些是 c++11 谷物库的一部分。我正在尝试自定义。

4

3 回答 3

16

以下可能会有所帮助:

template<typename T> struct is_shared_ptr : std::false_type {};
template<typename T> struct is_shared_ptr<std::shared_ptr<T>> : std::true_type {};

那么您可以执行以下操作以获得正确的功能:

template <class T> 
typename std::enable_if<is_shared_ptr<decltype(std::declval<T>().value)>::value, void>::type
func( T t )
{
    std::cout << "shared ptr" << std::endl;
}

template <class T> 
typename std::enable_if<!is_shared_ptr<decltype(std::declval<T>().value)>::value, void>::type
func( T t )
{
    std::cout << "non shared" << std::endl;
}

现场演示

于 2017-01-25T13:49:11.400 回答
10

这是模板专业化的一个基本案例。以下是确定类型 T 是否为 a 的类型特征shared_ptr。它可以以std::is_pointer您已经使用的相同方式使用。

#include <memory>
#include <type_traits>

template<class T>
struct is_shared_ptr : std::false_type {};

template<class T>
struct is_shared_ptr<std::shared_ptr<T>> : std::true_type {};

示范:

static_assert(is_shared_ptr<std::shared_ptr<int>>::value == true, "");
static_assert(is_shared_ptr<int>::value == false, "");
于 2017-01-25T13:51:05.483 回答
2

如果提供的类型本身是某个未知类型 T 的 std::shared_ptr,那么以下使用 SFINAE 应该会有所帮助!由于所有智能指针都提供成员类型“ element_type”,因此我们可以专门用于std::shared_ptr<T>std::weak_ptr<T>std::unique_ptr<T>,如下所示:

template<typename T, typename Enable = void>
struct is_smart_pointer
{
    enum { value = false };
};

template<typename T>
struct is_smart_pointer<T, typename std::enable_if<std::is_same<typename std::remove_cv<T>::type, std::shared_ptr<typename T::element_type>>::value>::type>
{
    enum { value = true };
};

template<typename T>
struct is_smart_pointer<T, typename std::enable_if<std::is_same<typename std::remove_cv<T>::type, std::unique_ptr<typename T::element_type>>::value>::type>
{
    enum { value = true };
};

template<typename T>
struct is_smart_pointer<T, typename std::enable_if<std::is_same<typename std::remove_cv<T>::type, std::weak_ptr<typename T::element_type>>::value>::type>
{
    enum { value = true };
};
于 2019-05-20T17:51:54.977 回答