当前位置:首页 > > 充电吧
[导读]       template 声明式中,class 和 typename 这两个关键字意义完全相同templateclass Widget; templateclass Widget;       

       template 声明式中,class 和 typename 这两个关键字意义完全相同

templateclass Widget;
templateclass Widget;

       有时候你一定要用 typename。

templatevoid print2nd(const C& container) 
{ 
    if (container.size() >= 2) 
    { 
        C::const_iterator iter(container.begin()); 
        ++iter; 
        int value = *iter; 
        std::cout << value; 
    } 
}

       iter 的类型是C::const_iterator实际上是什么必须取决于template 参数C。template内出现的名称如果相依于某个template参数,称之为从属名称(dependent names)。如果从属名称在class内呈嵌套状,称之为嵌套从属名称(nested dependent name)。C::const_iterator就是这样一个名称嵌套从属名称。value类型int,不依赖任何template参数的名称,称为非从属名称(non-dependent name)。
       嵌套从属名称可能导致解析的困难:

templatevoid print2nd(const C& container) 
{ 
    C::const_iterator* x; 
}

       看起来我们好像声明一个local变量,是个指针,指向一个C::const_iterator。 但它之所以被那么认为,是因为我们 “已经知道”C::const_iterator是个类型。如果C::const_iterator不是个类型呢?如果C有个static成员变量碰巧被命名为const_iterator。此时x碰巧是个global变量名称,那样上述代码就是一个相乘动作,C::const_iterator乘以x。撰写C++解析器的人必须操心所有可能的输入。
       在我们知道C以前,没有任何办法可以知道C::const_iterator是否为一个类型。而当编译器开始解析 template print2nd 时,尚未确定C是什么东西。
C++有个规则可以解析这一歧义状态:如果解析器在template 中遭遇一个嵌套从属名称,它便假设这个名称不是个类型,除非你告诉它是。缺省情况下从属名称不是类型。此规则还有个例外,后面会提到。
       所以上述代码不是有效的C++代码。我们必须告诉 C++ 说C::const_iterator是个类型。只要紧邻它之前放置关键字typename即可:

template//这个合法的C++代码 
void print2nd(const C& container) 
{ 
    if (container.size() >= 2) 
    { 
        typename C::const_iterator iter(container.begin()); 
        ++iter; 
        int value = *iter; 
        std::cout << value; 
    } 
}

       typename只用来验明嵌套从属类型名称,其他名称不该有它存在。

templatevoid f(const C& container,        // 不允许使用 typename 
       typename C::iterator iter);// 一定要使用 typename

       “typename必须作为嵌套从属类型名称的前缀词” 这一规则的例外是,typename 不可以出现在base classes list内的嵌套从属类型名称之前,也不可在member initialization list(成员初始化列表)中作为base class修饰符。例如:

templateclass Derived: public Base::Nested{// base class list中不允许“typename” 
public: 
    explicit Derived(int x) 
        :Base::Nested(x)           //mem.init.list中不允许“typename” 
    { 
       typename Base::Nested temp; //嵌套从属类型既不在base class list中也不在mem.init.list中, 
    }                                 // 作为一个base class修饰符需加上typename 
};

       让我们看一个typename例子:一个function template,他接受一个迭代器,而我们打算为该迭代器指涉的对象做一份复件temp:

templatevoid workWithIterator(IterT) 
{ 
    typename std::iterator_traits::value_type temp(*iter); 
}

       如果IterT是 vector

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

其实在 c++语言里面const修饰的才算是一个真正的常量,在 c 语言中 const 可以说是个“冒牌货”。为什么会这样?其实是 c++ 编译器对 const 进行了加强,当 c++ 编译器遇到常量声明时,不会像 c...

关键字: c++ C语言 const

返回函数的引用去初始化一个新的引用这个和前面一样,都是不会产生副本,但是现在是用返回值去初始化一个引用声明c,也就是说这时候变成了变量temp的别名,在c的生命周期内temp是一直有效的,这样做完全可以。

关键字: c++ 返回值 引用声明

C++是一种面向对象的高级程序设计语言,是C语言的超集。

关键字: c++ C语言

摘要:随着互联网技术与数据库技术的不断发展和相互结合,越来越多的用户需要访问在线数据库来获取各种信息,在该过程中,用戶要对数据库中的数据进行查询。因此,用户就必需要掌握一定的结构化查询语言SQL,而且还要对数据库模式有一...

关键字: 查询 SQL 模式图 关键字 简化子树

用一个简单的c程序例子,介绍c语言的基本构成、格式、以及良好的书写风格,使小伙伴对c语言有个初步认识。

关键字: C语言 标识符 关键字

今天来聊 final 关键字,因为最近在看的几本书都讲到了 final 关键字,发现好多小细节自己都忽视了,抽空总结了一下,分享给大家。

关键字: final 关键字

很多人遇到指针就糊涂,搞不清到底指向什么,其实是你没搞清楚 * 修饰谁,还有一些关键字修饰谁。

关键字: 指针 关键字

分析:这是Adobe 公司2007 年校园招聘的最新笔试题。这道题除了考察应聘者的C++ 基本功底外,还能考察反应能力,是一道很好的题目。 在Java 中定义了关键字final ,被final 修饰的

关键字: c++ class

纯虚函数在基类中是没有定义的,必须在子类中加以实现,很像java中的接口函数! 虚函数 引入原因:为了方便使用多态特性,我们常常需要在基类中定义虚函数。 class Cman { public: v

关键字: class 多态

泛型算法中的定制操作很多算法都会比较输入序列中的元素,通过定制比较动作,可以控制算法按照编程者的意图工作。本文以string排序为例进行说明,首先是缺省的排序动作: vector v{"This","

关键字: c++
关闭