条款 42:了解 typename 的双重意义

Effective C++ 中文版 第三版》读书笔记

条款 42:了解 typename 的双重意义

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

template<class T> class Widget;
template<typename T> class Widget;

有时候你一定要用 typename,

可以在 template 中指涉的两种名称:

template <typename C> 
void 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)。

嵌套从属名称可能导致解析的困难:

template <typename C> 
void 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 <typename C> //这个合法的 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 只用来验明嵌套从属类型名称;其他名称不该有它存在。

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

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

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

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

template <typename IterT> 
void workWithIterator(IterT) 
{ 
    typename std::iterator_traits<IterT>::value_type temp(*iter); 
}

这是个标准 trait class 的一种运用(条款 47),相当于说 “类型 IterT 之对象所指之物的类型”。如果 IterT 是 vector<int>::iterator,temp 的类型就是 int,如果 IterT 是 list<string>::iterator,temp 的类型就是 string。由于 std::iterator_traits<IterT>::value_type 是个嵌套从属类型名称(value_type 被嵌套于 iterator_traits<IterT> 之内而 IterT 是个 template 参数),所以必须在它之前放置 typename。

这么长你肯定会想建立一个 typedef。对于 traits 成员名称如 value_type,普遍习惯是设定 typedef 名称用以代表某个 traits 成员名称:

template <typename IterT>
void workWithIterator(IterT)
{
typedef typename std::iterator_traits<IterT>::value_type value_type;
value_type temp(*iter);
}

请记住:

  1. 声明 template 参数时,前缀关键字 class 和 typename 可互换。

  2. 请使用关键字 typename 标识嵌套从属类型名称;但不得在 base class list(基类列表)或 member initialization list(成员初值列表)内以它作为 base class 修饰符。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容

  • 提一个问题:以下template声明式中,class和typename有什么不同? 答案:没有不同。当我们声明te...
    何幻阅读 1,535评论 0 0
  • 再读高效c++,颇有收获,现将高效c++中的经典分享如下,希望对你有所帮助。 1、尽量以const \enum\i...
    橙小汁阅读 1,216评论 0 1
  • 本原则讨论的是typename,这个typename是个啥玩意呢,你经常会在泛型编程中见到如下代码: 这俩是在一般...
    Stroman阅读 592评论 0 0
  • 这里把STL里处理iterator的tag-dispatching + trait class机制提取一点出来并浅...
    Quasars阅读 527评论 0 1
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,647评论 18 139