当前位置: 首页 > 编程笔记 >

C++的template模板中class与typename关键字的区别分析

班浩皛
2023-03-14
本文向大家介绍C++的template模板中class与typename关键字的区别分析,包括了C++的template模板中class与typename关键字的区别分析的使用技巧和注意事项,需要的朋友参考一下

在C++模板中,可以使用class或者typename来声明模板参数,那么这两个关键字有什么区别呢?

模板参数声明
对于模板参数声明,这两个参数没有区别,含义是一样的。

template class Simple;
template class Simple;

上面两行都是声明一个模板类Simple.

表明类型
假如我们有这样一段代码:

template


void add(const T &acontainer, T &sum)
{
  T::const_iterator iter = container.begin();
  for (; iter != container.end(); ++iter) {
    sum += *iter;
  }
}

iter的类型是T::const_iterator,这个类型依赖模板参数T。把依赖模板参数的名称称为依赖名称。当这个依赖名称又在一个类中时,称为嵌套依赖名称。相对的,称为非嵌套依赖名称。

嵌套依赖名称会导致编译器编译的困难,例如下面的代码:

template


void add(const T &container)
{
  T::const_iterator *x;
  ...
}

这看起来像声明一个变量x,它的类型为T::const_iterator *。但是编译器并不知道,也有可能类T中又一个static数据成员const_iterator,或者正好有一个html" target="_blank">全局变量x。这时上面的html" target="_blank">代码就变成乘法操作。这是因为 C++编译器在处理模板的时候,会将需要推导的数据类型暂时搁置,到运行时再确定。

当编译器遇到一个模板中的嵌套依赖名称时,编译器将它作为变量对待。因此,需要显示的告诉编译器,这就需要使用关键字typename。

template
void add(const T &container, T &sum)
{
  typename T::const_iterator iter = container.begin();
  for (; iter != container.end(); ++iter) {
    sum += *iter;
  }
}

因此,使用嵌套依赖的类型名称时,都需要使用typename指定它是一种类型。

例外
嵌套依赖名称在基类列表中,或者在成员初始化列表中时,不能使用typename。

template
class Drived: public Base::Nested { // 基类列表,不要使用typename
public:
  explicit Derived(int x): Base::Nested(x) { // 成员初始化列表,不要使用typename
    typename Base::Nested temp;
    ...
  }
  ...
};

另外一些注意点

1、嵌套从属名称(nested dependent names)
假如template内出现的名称如果依赖于某个模板参数,则称其为从属名称(dependent names),如果从属名称在class内呈嵌套状则称之为嵌套从属名称(nested dependent names)。
例如:

templaet <typename T>void myPrint(const T& t){
  t::const_iterator iter(t.begin());
}

假设模板参数列表中的参数表示一个容器类型,则我们知道t::const_iterator一个依赖模板参数并且在容器内部,所以t::const_iterator是一个嵌套从属名称。

在我们知道t是什么之前没有办法可以知道t::const_iterator是否是一个类型,因为有还可能是个静态(static)成员变量,考虑下面的例子:

template <typename T>void myPrint(const T& t){
  t::const_iterator * x;
}

如果const_iterator是t的静态成员变量,则上面的t::const_iterator * x;中的*表示乘法,如果是个类型则表示声明一个指向t::const_iterator类型的指针。
从而给编译器造成困惑(因为我们不知道t是什么)。

C++有个规定:当解析器在模板中遇到一个嵌套从属名称时便假定这个名称不是类型,除非你用关键字typename指定它是:

template <typename T>void myPrint(const T& t){
  typename t::const_iterator * x;   //这样便不会造成困惑了}

同理不仅在内部,在参数列表里也是:

template <typename T>void f(const T& t, typename T::const_iterator cit){
   //T不是嵌套从属名称,而T::const_iterator是,所以要在T::const_iterator前面加上typename    //....}

2、是嵌套从属名称但不用加typename的两种情况
基类列表(base list)和成员初始化列表(member initializaiton list)
 
template <typename T>class Derived: public Base<T>::Nested {         //基类列表中不允许使用typenamepublic:
  explicit Derived(int x): Base<T>::Nested(int x){  //初始化列表中不允许使用typename    typename Base<T>::Nested temp;         //嵌套从属名称(既不在基类列表中又不在初始化列表中)前面必须要加typename  }
}
 类似资料:
  • 本文向大家介绍Python中关键字is与==的区别简述,包括了Python中关键字is与==的区别简述的使用技巧和注意事项,需要的朋友参考一下 本文以简单示例分析了python中关键字is与 ==的区别,供大家参考一下。 首先说明一下Python学习中几个相关的小知识点。 Python中的对象包含三要素:id、type、value 其中:id用来唯一标识一个对象,type标识对象的类型,value

  • 本文向大家介绍C++中的模板template小结,包括了C++中的模板template小结的使用技巧和注意事项,需要的朋友参考一下 函数模板 我们可以把函数模板当做一种特殊的函数,里面的参数类型可以是任意类型,这样的话我们就可以减少重复定义,从而让这个函数模板自动适应不同的参数类型,也就是说函数可以适应多种类型的参数,例如double、int或者类什么的。 C++为了实现上面的功能,引入了temp

  • 本文向大家介绍C#中的out和ref关键字之间的区别,包括了C#中的out和ref关键字之间的区别的使用技巧和注意事项,需要的朋友参考一下 out关键字 out关键字用于将参数作为引用类型传递给方法,主要用于方法必须返回多个值时。ref关键字还用于将参数作为引用类型传递给方法,并且在要在方法中修改现有变量时使用。以下是C#中ref和out关键字的有效用法。 示例 输出结果 以下是ref和out关键

  • 我的理解正确吗?pop()删除并返回列表中的最后一项,因此可以用pop()实现“先进先出”结构,也可以用pop(0)del实现“先进先出”结构来删除指定索引的项。所以,我的问题是这两者之间有什么区别。如果这两者是相同的,为什么Python创建者要费心创建两者呢?只有pop()返回remove项而del不返回的区别吗?

  • 本文向大家介绍C#中Html.RenderPartial与Html.RenderAction的区别分析,包括了C#中Html.RenderPartial与Html.RenderAction的区别分析的使用技巧和注意事项,需要的朋友参考一下 本文较为详细的讲解了C#中Html.RenderPartial与Html.RenderAction的区别,具体分析如下: Html.RenderPartial与

  • 本文向大家介绍C#中String与string的区别分析,包括了C#中String与string的区别分析的使用技巧和注意事项,需要的朋友参考一下 本文实例展示了C#程序设计中String与string的区别,对于C#初学者来说有很好的参考借鉴价值。具体如下: 一、区别分析: String:类,System.String string:类型,变量 两者本质上没有任何区别,都是System.Stri