【问题标题】:Template detects if T is pointer or class模板检测 T 是指针还是类
【发布时间】:2015-08-13 21:21:45
【问题描述】:

考虑以下代码:

class MyClass
{
    ...
};

template <typename Object>
class List
{
public:

    void insert(const Object & x)
    {
        // call when Object is MyClass
    }

    void insert(const Object & x)
    {
        // call when Object is MyClass*
    }
}

int main()
{
    MyClass a;

    List<MyClass> lst;
    List<MyClass*> plst;

    lst.insert(a);
    plst.insert(new Myclass);

    return 0;
}

如何根据模板是类还是指针来告诉编译器调用不同的方法?

如何修复上面的代码?

【问题讨论】:

  • 请发MCVE
  • 更好的方法是重写你的代码,让它不关心它是在处理一个类还是一个指针。标准的list&lt;T&gt;vector&lt;T&gt; 类不在乎,为什么你的List&lt;T&gt;?它在做什么取决于模板参数是类类型?

标签: c++ templates segmentation-fault sfinae


【解决方案1】:

您可以使用std::is_pointerstd::enable_if 的组合:

#include <type_traits>
#include <iostream>

class MyClass
{
};

template <typename Object>
class List
{
public:

    template<class T=Object>
    void insert(T t, typename std::enable_if<std::is_pointer<T>::value >::type* = 0) 
    {
        std::cout << "insert pointer" << std::endl;
    }

    template<class T=Object>
    void insert(T t, typename std::enable_if<!std::is_pointer<T>::value >::type* = 0) 
    {
        std::cout << "insert non-pointer" << std::endl;
    }
};

int main()
{
    MyClass a;

    List<MyClass> lst;
    List<MyClass*> plst;

    lst.insert(a);
    plst.insert(new MyClass());

    return 0;
}

现场示例:https://ideone.com/CK8Zdo

这将允许您将指针和非指针都插入到指针或非指针列表中。 如果你想限制它,你可以使用这个:

#include <type_traits>
#include <iostream>
class MyClass
{
};

template <typename Object>
class List
{
public:

    template<class T=Object>
    void insert(T t, typename std::enable_if<std::is_same<T,Object>::value&&std::is_pointer<T>::value >::type* = 0) 
    {
        std::cout << "insert pointer" << std::endl;
    }

    template<class T=Object>
    void insert(const T& t, typename std::enable_if<std::is_same<T,Object>::value&&!std::is_pointer<T>::value >::type* = 0) 
    {
        std::cout << "insert non-pointer" << std::endl;
    }
};

int main()
{
    MyClass a;

    List<MyClass> lst;
    List<MyClass*> plst;

    lst.insert(a);
    // plst.insert(a); // compiler error

    // lst.insert(new MyClass()); // compiler error
    plst.insert(new MyClass());


    return 0;
}

现场示例:https://ideone.com/3DtBfr

【讨论】:

  • 我在尝试使用此方法时遇到编译错误。 error: failed requirement 'std::is_pointer&lt;MyType&gt;::value'; 'enable_if' cannot be used to disable this declaration
  • @einstein 哪个例子?哪个编译器?
  • 我只是忘记在您的示例中为方法中的类重新分配模板参数。但是这样做之后,效果很好!
【解决方案2】:

我知道我的回答并不完全符合您的要求,但也许会有所帮助。

我相信您的意图是让 List 类具有一个插入方法(不是两个),并且此方法的行为应取决于您的模板参数。为了这 您可以为指针编写类的特化。然后基本模板将用于非指针类型,专门化将用于指针类型。

您的代码如下所示:

template <typename Object>
class List
{
public:

    void insert(const Object & x)
    {
        // call when Object is MyClass
    }
};

template <typename Object>
class List<Object *>
{
public:

    void insert(Object * x)
    {
        // call when Object is MyClass*
    }
};

【讨论】:

    【解决方案3】:
    void insert(const Object & x)
    {
        M_insert(x, dispatcher<std::is_pointer<Object>::value> );
    }
    

    List 内部使用调度器

    template <bool B> class dispatcher {};
    using ObjectPtr   = dispatcher<true>;
    using ObjectValue = dispatcher<false>;
    

    然后发送到M_insert:

    void M_insert(const Object &p, ObjectPtr) { // Object is a pointer }
    void M_insert(const Object &p, ObjectValue) { // Object is not a pointer }
    

    实时示例here。但是,我鼓励您确定您是否真的需要它并可能相应地修复您的设计。

    【讨论】:

      【解决方案4】:

      这就是诀窍:

      template <typename Object>
      class List
      {
      public:
      
          template<class C = Object>
          void insert(const C & x)
          {
              // call when Object is MyClass
              std::cout << "1" << "\n" ;
          }
      
          template<class P = Object*>
             void insert(P* p)
          {
              // call when Object is MyClass*
              std::cout << "2" << "\n" ;
          }
      } ;
      

      Here 是一个工作示例。

      【讨论】:

      • 但这将允许插入我认为不允许的非Object 数据类型。
      猜你喜欢
      • 1970-01-01
      • 2017-06-25
      • 1970-01-01
      • 2023-03-10
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2016-11-23
      相关资源
      最近更新 更多