How to properly declare a self-referencing template type?

前端 未结 4 1865
误落风尘
误落风尘 2021-01-19 02:17

How do I declare a templated type that refers to itself?

template  class Animal
{
public:
    T getChild ();
}

With

相关标签:
4条回答
  • 2021-01-19 02:20

    In this case, you need to specify some type parameter to Animal in your typename definition, or else it would be an "infinite recursion" in the type construction:

    template<class T> class Animal;//you'll need this forward declaration
    
    template <class T = Animal<int> > class Animal //int is just an example
    {
    public:
        T getPrey ();
    }
    
    0 讨论(0)
  • 2021-01-19 02:22

    The usual way to do something like a linked list is:

    template <class T> class Linked
    {
    private:
        Linked<T>* m_prev;
        Linked<T>* m_next;
    }
    

    Does this work for you, and if not, what are you trying to accomplish that can't be done this way?

    0 讨论(0)
  • 2021-01-19 02:26

    The OP has been answered but I want to chime in because the immediate cause of the problem is not recursion, as others claim. The simplest reason this wouldn't work is that class templates are not types. They are templates. Similarly, function templates are not functions either. So all of this is nonsensical:

    template<typename T> int function_template(int);
    typedef int function_type(int);
    void eats_a_function(function_type&); // needs a reference to function
    
    std::vector< std::vector > vec0; // std::vector is not a type
    std::vector< std::list > vec1; // std::list is not a type
    eats_a_function(function_template); // function_template is not a function
    

    Notice that in the vec1 case, std::list is not related to std::vector. The template is fully defined (assuming header inclusion) at the point of instantiation. It still won't work.

    Instead, the following works:

    std::vector< std::vector<int> > vec2; // std::vector<int> is a type
    std::vector< std::list<double> > vec3; // std::list<double> is a type
    eats_a_function(function_template<long>); // function_template<long> is a function
    

    Notice that in the vec2 case, it's fine to pass an instantiation of the template itself.


    For the record, a toy solution to the toy problem on writing a template that refers to itself, using the proverbial layer of indirection:

    // expects a template that expects a type
    template<template<class> class T> struct indirection {};
    
    // forward decl. for defaulting the parameter
    template<typename T> struct recursive;
    
    // template that expects a type
    template<typename T = indirection<recursive> > struct recursive {};
    

    Not terribly powerful given the few things that are possible with a template (the T parameter inside indirection). It's of course possible to write a rebind-style metafunction that returns an instantiation of T.

    0 讨论(0)
  • 2021-01-19 02:29

    You can't create a template class wherein the template type is the class itself. This is a logical recursion that your compiler can't mitigate. Since templates require the compiler to construct the object when a specific typing is encountered ( say Animal ), you have to have a complete definition of the template type. Otherwise, the compiler will recursively try to construct objects. Since your object self-references, this will be an non-terminating recursion.

    This use-case seems much more appropriate for inheritance than for templates.

    Doing this will let you compile: template class Animal { public: T getPrey (); }

    Or, if you really wan't a default argument:

    template <class T=Cheetah> class Animal
    {
    public:
        T getPrey ();
    }
    

    Yet, cheetah must not use itself or Animal as one of its potential template types.

    0 讨论(0)
提交回复
热议问题