代码之家  ›  专栏  ›  技术社区  ›  Hani Gotc

模板中的typedef导致shadows template parm错误[重复]

  •  2
  • Hani Gotc  · 技术社区  · 6 年前

    我在模板类中使用typedef声明映射,如下所示:

    #include <map>
    
    template <typename T> class LocalStub {
      typedef std::map<T, T> QueryMap;
      typedef std::pair<T, T> QueryPair;
      typedef QueryMap::iterator QueryMapIterator;
    
    public:
      LocalStub();
      ~LocalStub();
      void AddQuery(const T &, const T &);
      const T &Answer(const T &) const;
    private:
      QueryMap _queryMap;
    };
    

    编译错误

    ../src/LocalStub/include/localstub.hpp:12: error: declaration of 'class T'
    ../src/LocalStub/include/localstub.hpp:11: error:  shadows template parm 'class T'
    ../src/LocalStub/include/localstub.hpp:13: error: template declaration of 'typedef'
    ../src/LocalStub/include/localstub.hpp:14: error: declaration of 'class T'
    ../src/LocalStub/include/localstub.hpp:11: error:  shadows template parm 'class T'
    ../src/LocalStub/include/localstub.hpp:15: error: template declaration of 'typedef'
    ../src/LocalStub/include/localstub.hpp:16: error: declaration of 'class T'
    ../src/LocalStub/include/localstub.hpp:11: error:  shadows template parm 'class T'
    ../src/LocalStub/include/localstub.hpp:17: error: template declaration of 'typedef'
    ../src/LocalStub/include/localstub.hpp:26: error: 'QueryMap' does not name a type
    

    我做错什么了?我不明白我为什么会犯这个错误。

    2 回复  |  直到 6 年前
        1
  •  2
  •   ConsistentProgrammer    6 年前

    你忘了写关键字了 typename 具有 QueryMapIterator . 以下是更新版本:

    template <typename T> class LocalStub {
      typedef std::map<T, T> QueryMap;
      typedef std::pair<T, T> QueryPair;
      typedef typename QueryMap::iterator QueryMapIterator;
    
    public:
      LocalStub();
      ~LocalStub();
      void AddQuery(const T &, const T &);
      const T &Answer(const T &) const;
    private:
      QueryMap _queryMap;
    };
    

    这一要求的原因是编译器目前不知道 查询映射迭代器 描述成员变量或嵌套类型

        2
  •  1
  •   bartop    6 年前

    TL;医生- typedef typename QueryMap::iterator QueryMapIterator;

    较长版本: QueryMap::iterator 是一个 dependant name 因此,必须使用 typename 之前 typedef 就这样。依赖名称问题来自于模板构造内部 type::something 这个 something 可以引用类型、值或函数,如下例所示:

    template<class T>
    struct foo{
        void bar(){ T::something; }
    };
    
    struct baz{
        using something = int;
    };
    
    struct bez{
        static const int something = 0;
    };
    

    因此 类别名 如果要传递的名称引用类型,则需要向编译器提供附加信息。