Effective C++筆記:template中的class、typename關(guān)鍵字
? ? ? ?template 聲明式中,class 和 typename 這兩個關(guān)鍵字意義完全相同
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 參數(shù)C。template內(nèi)出現(xiàn)的名稱如果相依于某個template參數(shù),稱之為從屬名稱(dependent names)。如果從屬名稱在class內(nèi)呈嵌套狀,稱之為嵌套從屬名稱(nested dependent name)。C::const_iterator就是這樣一個名稱嵌套從屬名稱。value類型int,不依賴任何template參數(shù)的名稱,稱為非從屬名稱(non-dependent name)。
? ? ? ?嵌套從屬名稱可能導(dǎo)致解析的困難:
templatevoid?print2nd(const?C&?container)? {? ????C::const_iterator*?x;? }
? ? ? ?看起來我們好像聲明一個local變量,是個指針,指向一個C::const_iterator。 但它之所以被那么認(rèn)為,是因為我們 “已經(jīng)知道”C::const_iterator是個類型。如果C::const_iterator不是個類型呢?如果C有個static成員變量碰巧被命名為const_iterator。此時x碰巧是個global變量名稱,那樣上述代碼就是一個相乘動作,C::const_iterator乘以x。撰寫C++解析器的人必須操心所有可能的輸入。
? ? ? ?在我們知道C以前,沒有任何辦法可以知道C::const_iterator是否為一個類型。而當(dāng)編譯器開始解析 template print2nd 時,尚未確定C是什么東西。
C++有個規(guī)則可以解析這一歧義狀態(tài):如果解析器在template 中遭遇一個嵌套從屬名稱,它便假設(shè)這個名稱不是個類型,除非你告訴它是。缺省情況下從屬名稱不是類型。此規(guī)則還有個例外,后面會提到。
? ? ? ?所以上述代碼不是有效的C++代碼。我們必須告訴 C++ 說C::const_iterator是個類型。只要緊鄰它之前放置關(guān)鍵字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必須作為嵌套從屬類型名稱的前綴詞” 這一規(guī)則的例外是,typename 不可以出現(xiàn)在base classes list內(nèi)的嵌套從屬類型名稱之前,也不可在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,他接受一個迭代器,而我們打算為該迭代器指涉的對象做一份復(fù)件temp:
templatevoid?workWithIterator(IterT)? {? ????typename?std::iterator_traits::value_type?temp(*iter);? }
? ? ? ?如果IterT是 vector