C++ åèæå
- C++11
- C++14
- C++17
- C++20
- C++ ç¼è¯å¨æ¯ææ åµè¡¨
- ç¬ç«ä¸å®¿ä¸»å®ç°
- C++ è¯è¨
- å鿍¡æ¿(C++14 èµ·)
- æ´æ°åé¢é
- èååå§å
- æ¯è¾è¿ç®ç¬¦
- é»è®¤æ¯è¾(C++20 èµ·)
- 转ä¹åºå
- for 循ç¯
- while 循ç¯
- ç¨æ·å®ä¹è½¬æ¢
- SFINAE
- 䏻彿°
- ASCII ç 表
- æ è¯ç¬¦
- ç±»å
- å 忍¡å
- 对象
- åºæ¬æ¦å¿µ
- 表达å¼
- 声æ
- åå§å
- 彿°
- è¯å¥
- ç±»
- è¿ç®ç¬¦éè½½
- 模æ¿
- å¼å¸¸
- äºå¡æ§å å
- å ä½ç¬¦ç±»å说æç¬¦ (C++11 èµ·)
- decltype 说æç¬¦
- 彿°å£°æ
- final 说æç¬¦ (C++11 èµ·)
- override 说æç¬¦(C++11 èµ·)
- å¼ç¨å£°æ
- ç§»å¨æé 彿°
- ç§»å¨èµå¼è¿ç®ç¬¦
- æä¸¾å£°æ
- constexpr 说æç¬¦(C++11 èµ·)
- å表åå§å (C++11 èµ·)
- æé 彿°ä¸æååå§åå¨å表
- using 声æ
- nullptrï¼æéåé¢é
- åºç¡ç±»å
- ç±»åå«åï¼å«å模ç (C++11 èµ·)
- å½¢åå
- èåä½å£°æ
- å符串åé¢é
- ç¨æ·å®ä¹åé¢é (C++11 èµ·)
- 屿§è¯´æç¬¦åºå(C++11 èµ·)
- Lambda è¡¨è¾¾å¼ (C++11 èµ·)
- noexcept 说æç¬¦ (C++11 èµ·)
- noexcept è¿ç®ç¬¦ (C++11 èµ·)
- alignof è¿ç®ç¬¦(C++11 èµ·)
- alignas 说æç¬¦ (C++11 èµ·)
- åå¨ç±»è¯´æç¬¦
- åºäºèå´ç for å¾ªç¯ (C++11 èµ·)
- static_assert 声æ
- éå¼è½¬æ¢
- 代ç¨è¿ç®ç¬¦è¡¨ç¤º
- èªå¢/èªåè¿ç®ç¬¦
- æå 表达å¼(C++17 èµ·)
- 类模æ¿å®åæ¨å¯¼(C++17 èµ·)
- 模æ¿å½¢å䏿¨¡æ¿å®å
- if è¯å¥
- inline 说æç¬¦
- ç»æåç»å®å£°æ (C++17 èµ·)
- switch è¯å¥
- å符åé¢é
- å½å空é´
- æ±å¼é¡ºåº
- å¤å¶æ¶é¤
- consteval 说æç¬¦ (C++20 èµ·)
- constinit 说æç¬¦ (C++20 èµ·)
- åç¨ (C++20)
- 模å (C++20 èµ·)
- 约æä¸æ¦å¿µ (C++20 èµ·)
- new 表达å¼
- do-while 循ç¯
- continue è¯å¥
- break è¯å¥
- goto è¯å¥
- return è¯å¥
- 卿å¼å¸¸è¯´æ
- throw 表达å¼
- try å
- å½å空é´å«å
- 类声æ
- cvï¼const ä¸ volatileï¼ç±»åéå®ç¬¦
- é»è®¤åå§å
- å¼åå§å(C++03 èµ·)
- é¶åå§å
- å¤å¶åå§å
- ç´æ¥åå§å
- 常éåå§å
- å¼ç¨åå§å
- å¼ç±»å«
- C++ è¿ç®ç¬¦ä¼å 级
- å¸å°åé¢é
- æµ®ç¹åé¢é
- typedef 说æç¬¦
- æ¾å¼ç±»å转æ¢
- static_cast 转æ¢
- dynamic_cast 转æ¢
- const_cast 转æ¢
- reinterpret_cast 转æ¢
- delete 表达å¼
- æé 彿°ä¸æååå§åå¨å表
- this æé
- 访é®è¯´æç¬¦
- åå 声æ
- virtual 彿°è¯´æç¬¦
- explicit 说æç¬¦
- éææå
- é»è®¤æé 彿°
- å¤å¶æé 彿°
- å¤å¶èµå¼è¿ç®ç¬¦
- ææå½æ°
- 类模æ¿
- 彿°æ¨¡æ¿
- æ¾å¼ï¼å ¨ï¼æ¨¡æ¿ç¹å
- æ±ç¼å£°æ
- C++ çåå²
- ä½ç¨å
- çåæ
- å®ä¹ä¸åä¸å®ä¹è§åï¼ODRï¼
- å忥æ¾
- æéå®çå忥æ¾
- æ éå®çå忥æ¾
- å¦åè§å
- æªå®ä¹è¡ä¸º
- ç¿»è¯é¶æ®µ
- 常é表达å¼
- èµå¼è¿ç®ç¬¦
- ç®æ¯è¿ç®ç¬¦
- é»è¾è¿ç®ç¬¦
- æå访é®è¿ç®ç¬¦
- å ¶ä»è¿ç®ç¬¦
- sizeof è¿ç®ç¬¦
- typeid è¿ç®ç¬¦
- æé声æ
- æ°ç»å£°æ
- è¯è¨é¾æ¥
- 详述类å说æç¬¦
- é»è®¤å®å
- åé¿å®å
- å®åä¾èµæ¥æ¾
- éè½½å³è®®
- éè½½å½æ°çå°å
- æ³¨å ¥ç±»å
- ééææ°æ®æå
- ééææå彿°
- åµå¥ç±»
- æ´¾çç±»
- 空åºç±»ä¼å
- æ½è±¡ç±»
- ä½å
- è½¬æ¢æé 彿°
- æå模æ¿
- 模æ¿å®åæ¨å¯¼
- é¨å模æ¿ç¹å
- sizeof... è¿ç®ç¬¦
- å¾ å³å
- 彿° try å
- æ©å å½åç©ºé´ std
- åæ¯ç¼©å
- RAII
- ä¸/äº/é¶ä¹æ³å
- PImpl
- é¶å¼éåå
- ç±»å
- éå¼è½¬æ¢
- 注é
- C++ å ³é®è¯
- é¢å¤çå¨
- C++ æ ååºå¤´æä»¶
- å ·åè¦æ±
- åè½ç¹æ§æµè¯ (C++20)
- å·¥å ·åº
- ç±»åæ¯æï¼åºæ¬ç±»åãRTTIãç±»åç¹æ§ï¼
- æ¦å¿µåº (C++20)
- é误å¤ç
- 卿å å管ç
- æ¥æåæ¶é´å·¥å ·
- å符串åº
- 容å¨åº
- è¿ä»£å¨åº
- èå´åº (C++20)
- ç®æ³åº
- æ°å¼åº
- è¾å ¥/è¾åºåº
- æä»¶ç³»ç»åº
- æ¬å°ååº
- æ£å表达å¼åº
- ååæä½åº
- çº¿ç¨æ¯æåº
- å®éªæ§ C++ ç¹æ§
- æç¨çèµæº
- ç´¢å¼
- std 符å·ç´¢å¼
- åç¨æ¯æ (C++20)
- C++ å ³é®è¯
ä½ç½®ï¼é¦é¡µ > C++ åèæå >C++ è¯è¨ > å¾ å³å
å¾ å³å
卿¨¡æ¿ï¼ç±»æ¨¡æ¿å彿°æ¨¡æ¿ï¼å®ä¹ä¸ï¼æäºæé çå«ä¹å¯ä»¥å¨ä¸åçå®ä¾åé´ææä¸åãç¹å«æ¯ï¼ç±»åå表达å¼å¯ä»¥åå³äºç±»å模æ¿å½¢åçç±»åï¼åéç±»åæ¨¡æ¿å½¢åçå¼ã
template<typename T> struct X : B<T> // "B<T>" åå³äº T { typename T::A* pa; // "T::A" åå³äº T // ï¼æ¤ "typename" ç使ç¨è§ä¸æï¼ void f(B<T>* pb) { static int i = B<T>::i; // "B<T>::i" åå³äº T pb->j++; // "pb->j" åå³äº T } };
å¯¹å¾ å³ååéå¾ å³åçå忥æ¾åç»å®ææä¸åã
ç»å®è§å
éå¾ å³å卿¨¡æ¿å®ä¹ç¹æ¥æ¾å¹¶ç»å®ãå³ä½¿å¨æ¨¡æ¿å®ä¾åç¹ææ´å¥½çå¹é ï¼ä¹ä¿ææ¤ç»å®ï¼
è¥éå¾ å³åçå«ä¹å¨å®ä¹ç¹å模æ¿ç¹åçå®ä¾åç¹é´æææ¹åï¼åç¨åºéè¯æï¼ä¸è¦æ±è¯æãè¿å¨ä¸åæ å½¢å¯è½åºç°ï¼
- ç¨äºéå¾ å³åçç±»åå¨å®ä¹ç¹ä¸å®æ´ä½å¨å®ä¾åç¹å®æ´
- (C++17) å®ä¾å使ç¨äºå¨å®ä¹ç¹å°æªå®ä¹çé»è®¤å®åæé»è®¤æ¨¡æ¿å®å
- (C++17) å®ä¾åç¹çæä¸ªå¸¸é表达å¼ä½¿ç¨äºæ´åææ ä½ç¨åæä¸¾ç±»åç const 对象çå¼ï¼constexpr 对象çå¼ï¼å¼ç¨çå¼ï¼æè constexpr 彿°çå®ä¹ï¼è该对象/å¼ç¨/彿°å¨æ¨¡æ¿çå®ä¹ç¹å°æªå®ä¹
- (C++17) 该模æ¿å¨å®ä¾åç¹ä½¿ç¨äºéå¾ å³ç类模æ¿ç¹åæå鿍¡æ¿ç¹åï¼è宿ç¨çè¿ä¸ªæ¨¡æ¿ï¼æè ä»æä¸ªäºå®ä¹ç¹å¤å°æªå®ä¹çé¨åç¹åæå®ä¾åï¼æè æåäºæä¸ªäºå®ä¹ç¹å¤å°æªå£°æçæ¾å¼ç¹åã
å¾ å³åçç»å®åå»¶è¿å°æ¥æ¾åçæ¶ã
æ¥æ¾è§å
å¦å忥æ¾ä¸æè®¨è®ºçï¼å¯¹äºæ¨¡æ¿ä¸æä½¿ç¨çå¾ å³åçæ¥æ¾å»¶è¿å°æ¨¡æ¿å®åå·²ç¥æ¶ï¼å±æ¶
- é ADL æ¥æ¾æ£éªä»æ¨¡æ¿å®ä¹çè¯å¢å¯è§çå ·æå¤é¨è¿æ¥ç彿°å£°æ
- ADL ä¸å¹¶æ£éªä»æ¨¡æ¿å®ä¹çè¯å¢æä»æ¨¡æ¿å®ä¾åçè¯å¢å¯è§çå ·æå¤é¨è¿æ¥ç彿°å£°æ
ï¼æ¢è¨ä¹ï¼å¨æ¨¡æ¿å®ä¹åæ·»å æ°å½æ°å£°æä¸ä¼ä»¤å ¶å¯è§ï¼é¤ééè¿ ADLï¼ã
æ¤è§åçç®çæ¯å¸®å©æ¨¡æ¿å®ä¾åæµå¾¡ ODR è¿è§ï¼
// æä¸ªå¤é¨åº namespace E { template<typename T> void writeObject(const T& t) { std::cout << "Value = " << t << '\n'; } }  // ç¿»è¯åå 1ï¼ // ç¨åºå 1 叿å 许 E::writeObject ä¸ vector<int> ä¸åä½¿ç¨ namespace P1 { std::ostream& operator<<(std::ostream& os, const std::vector<int>& v) { for(int n: v) os << n << ' '; return os; } void doSomething() { std::vector<int> v; E::writeObject(v); // éè¯¯ï¼æ¾ä¸å° P1::operator<< } }  // ç¿»è¯åå 2ï¼ // ç¨åºå 2 叿å 许 E::writeObject ä¸ vector<int> ä¸åä½¿ç¨ namespace P2 { std::ostream& operator<<(std::ostream& os, const std::vector<int>& v) { for(int n: v) os << n <<':'; return os << "[]"; } void doSomethingElse() { std::vector<int> v; E::writeObject(v); // éè¯¯ï¼æ¾ä¸å° P2::operator<< } }
å¨ä¸ä¾ä¸ï¼è¥å 许ä»å®ä¾åè¯å¢è¿è¡é ADL æ¥æ¾ï¼å E::writeObject<vector<int>> çå®ä¾å伿¥æä¸¤ä¸ªä¸åå®ä¹ï¼ä¸ä¸ªä½¿ç¨ P1::operator<<ï¼ä¸ä¸ªä½¿ç¨ P2::operator<<ãè¿æ¥å¨å¯è½æ æ³æ£æµè¿ç§ ODR è¿è§ï¼å¯¼è´ä¸¤ä¸ªå®ä¾ä¸é½ä½¿ç¨å ¶ä¸ä¸ä¸ªæå¦ä¸ä¸ªã
ä¸ºä½¿å¾ ADL æ£æµç¨æ·å®ä¹å½å空é´ï¼è¦ä¹åºå½å° std::vector æ¿æ¢æç¨æ·å®ä¹ç±»ï¼è¦ä¹å
¶å
ç´ ç±»ååºå½ä¸ºç¨æ·å®ä¹ç±»ï¼
namespace P1 { // è¥ C æ¯å®ä¹äº P1 å½å空é´ä¸çç±» std::ostream& operator<<(std::ostream& os, const std::vector<C>& v) { for(C n: v) os << n; return os; } void doSomething() { std::vector<C> v; E::writeObject(v); // OKï¼å®ä¾å writeObject(std::vector<P1::C>) // éè¿ ADL æ¾å° P1::operator<< } }
注æï¼æ¤è§å使å¾ä¸ºæ ååºç±»åéè½½è¿ç®ç¬¦ä¸éäºå®è·µ
#include <iostream> #include <vector> #include <iterator> #include <utility>  // å主æï¼å ¨å±å½å空é´ä¸çè¿ç®ç¬¦ï¼ä½å ¶å®åå¤äº std:: ä¸ std::ostream& operator<<(std::ostream& os, std::pair<int, double> p) { return os << p.first << ',' << p.second; }  int main() { typedef std::pair<int, double> elem_t; std::vector<elem_t> v(10); std::cout << v[0] << '\n'; // OKï¼æ®éæ¥æ¾æ¾å°Â ::operator<< std::copy(v.begin(), v.end(), std::ostream_iterator<elem_t>(std::cout, " ")); // é误ï¼ä» // std::ostream_iterator å®ä¹ç¹çæ®éæ¥æ¾å ADL å°åªèè std å½å空é´ï¼ // èä¸å°æ¾å° std::operator<< çå¤ä¸ªéè½½ï¼æ è¿ç§æ¥æ¾ä¼å®æã // ä¹å卿¥æ¾ææ¾å°çéåä¸ï¼éè½½å³è®®ä¸º elem_t å¯»æ¾ operator<< ä¼å¤±è´¥ã }
注æï¼å¨æ¨¡æ¿å®ä¹æ¶ï¼ä¹ä¼åçå¾ å³åçæéå¶çæ¥æ¾ï¼ä½éç»å®ï¼ï¼ä»¥ä¾¿å°å®ä»¬ä¸éå¾ å³åè¿è¡åºåï¼ä¹ç¨äºç¡®å®å®ä»¬æ¯å½åå®ä¾åçæåè¿æ¯æªç¥ç¹åçæåãæ¤æ¥æ¾æè·å¾çä¿¡æ¯å¯ç¨äºæ£æµé误ï¼è§ä¸æã
å¾ å³ç±»å
ä¸åç±»åæ¯å¾ å³ç±»åï¼
- 模æ¿å½¢å
- æªç¥ç¹åï¼è§ä¸æï¼çæå
- ä½ä¸ºæªç¥ç¹åï¼è§ä¸æï¼çå¾ å³æåçåµå¥ç±»/æä¸¾
- å¾ å³ç±»åç cv éå®çæ¬
- ä»å¾ å³ç±»åææçå¤åç±»å
- å ç´ ç±»åå¾ å³æå ¶è¾¹çï¼è¥åå¨ï¼ä¸ºå¼å¾ å³çæ°ç»ç±»å
|
(C++17 èµ·) |
- æ¨¡æ¿æ è¯ï¼å ¶ä¸
- 模æ¿åæ¯æä¸ªæ¨¡æ¿å½¢åï¼æè
- 任使¨¡æ¿å®å为类åå¾ å³çï¼å¼å¾ å³çï¼æä¸ºå å±å¼ï¼å³ä½¿ä¸å¸¦å ¶å®ååè¡¨ä½¿ç¨æ¨¡æ¿æ è¯ï¼å¦æ³¨å ¥ç±»åï¼ä¹æ¯å¦æ¤ï¼
- åºç¨å°ç±»åå¾ å³è¡¨è¾¾å¼ç decltype çç»æ
注æï¼å½åå®ä¾åç typedef æåï¼ä» å½å ¶æä»£çç±»åå¾ å³æ¶æä¸ºå¾ å³çã
ç±»åå¾ å³è¡¨è¾¾å¼
ä¸å表达å¼ä¸ºç±»åå¾ å³çï¼
- ä»»ä½å表达å¼ä¸ºç±»åå¾ å³è¡¨è¾¾å¼ç表达å¼
- thisï¼è¥å ¶ç±»ä¸ºå¾ å³ç±»åã
- æ è¯è¡¨è¾¾å¼ï¼å ¶
- å å«æä¸ªæ è¯ç¬¦ï¼å ¶ååæ¥æ¾æ¾å°è³å°ä¸ä¸ªå¾ å³å£°æ
- å å«å¾ å³çæ¨¡æ¿æ è¯
- å
å«ç¹æ®æ è¯ç¬¦
__func__ï¼è¥æä¸ªå¤å´å½æ°æ¯æ¨¡æ¿ï¼ç±»æ¨¡æ¿ç鿍¡æ¿æåï¼ææ³å lambda (C++14 èµ·)ï¼ - å å«å°æä¸ªå¾ å³ç±»åç转æ¢å½æ°å
- å å«ä½ä¸ºæªç¥ç¹åæåçåµå¥å说æç¬¦ææé宿 è¯
- æåå½åå®ä¾åçæä¸ªå¾ 峿åï¼å®æ¯âæªç¥è¾¹ççæ°ç»âç±»åçéææ°æ®æå
|
(C++14 èµ·) |
|
(C++17 èµ·) |
- å°ä»»ä½å¾ å³ç±»åç转å表达å¼
- åå»ºå¾ å³ç±»å对象ç new 表达å¼
- æä»£å½åå®ä¾åçæä¸ªç±»åå¾ å³çæåçæå访é®è¡¨è¾¾å¼
- æä»£æªç¥ç¹åçæä¸ªæåçæå访é®è¡¨è¾¾å¼
| (C++17 èµ·) |
注æï¼åé¢éã伪ææå½æ°è°ç¨ãsizeofãalignofãtypeidãdelete 表达å¼ãthrow 表达å¼å noexcept 表达å¼å§ç»ä¸æ¯ç±»åå¾
å³çï¼å 为è¿äºè¡¨è¾¾å¼çç±»åä¸å¯è½å¾
å³ã
å¼å¾ å³è¡¨è¾¾å¼
- å¨è¦æ±å¸¸é表达å¼çè¯å¢ä¸ä½¿ç¨ç表达å¼ï¼ä¸å ¶æä»»ä½åè¡¨è¾¾å¼æ¯å¼å¾ å³ç
- æ è¯è¡¨è¾¾å¼ï¼å ¶
- 为类åå¾ å³ç
- 为æä¸ªéç±»åæ¨¡æ¿å½¢åçåå
|
(C++14 èµ·) |
- ä¸ºå ·æåé¢ç±»åç常éï¼å¹¶ä»å¼å¾ å³è¡¨è¾¾å¼åå§å
- sizeofãalignofãtypeidãnoexcept 表达å¼ï¼å ¶å®åæ¯ç±»åå¾ å³è¡¨è¾¾å¼æå¾ å³çç±»åæ è¯
- ä»»ä½åå¾ å³ç±»åè½¬æ¢æä»å¼å¾ å³è¡¨è¾¾å¼è½¬æ¢ç转å表达å¼
|
(C++14 èµ·) |
| (C++17 èµ·) |
å¾ å³å
| æ¬èæªå®æ åå ï¼æ¥èª [temp.dep] çé¢å¥ï¼ç°å¨ç¼ºå¤±ï¼åéææ¬å·å表ç id 表达å¼â¦â¦ï¼ |
| æ¬èæªå®æ åå ï¼éæ°é£è¯ä»¥ä»¤å®æ´æ¸ æ°ï¼æè³å°ä¸è¦é£ä¹å人ï¼ï¼å¹¶å¨ ait æ¶åºç¨ CWG é®é¢ 591 |
å½åå®ä¾å
å¨ç±»æ¨¡æ¿å®ä¹å
ï¼å
å«å
¶æå彿°ååµå¥ç±»ï¼ï¼ä¸äºåå坿¨å¯¼ä¸ºæä»£å½åå®ä¾åãè¿å
许å¨å®ä¹ç¹ï¼èéå¨å®ä¾åç¹æ£æµæäºé误ï¼å¹¶ç§»é¤å¯¹å¾
å³åä¸ç typename å template æ¶æ§ä¹ç¬¦çè¦æ±ï¼è§ä¸æã
ä» ä¸åååå¯ä»¥æä»£å½åå®ä¾åï¼
- å¨ç±»æ¨¡æ¿ç¹åä¸ï¼
- åµå¥ç±»ãç±»æ¨¡æ¿æåãåµå¥ç±»æåãæ¨¡æ¿çæ³¨å ¥ç±»åãåµå¥ç±»çæ³¨å ¥ç±»å
- å¨ä¸»æ¨¡æ¿çå®ä¹æå¨å ¶æåçå®ä¹ä¸ï¼
- 类模æ¿çåååé主模æ¿çå®ååè¡¨ï¼æçä»·çå«å模æ¿ç¹åï¼ï¼å ¶ä¸æ¯ä¸ªå®åçä»·äºå ¶å¯¹åºå½¢åãæ³¨æè¥è¡¨è¾¾å¼ç¨ä½éç±»åæ¨¡æ¿å®åï¼ä¾å¦ï¼N+0ï¼å ¶ä¸ N æ¯å½¢åï¼ï¼åå®ä¸æåå½åå®ä¾åï¼å³ä½¿å ¶å¼å¹é ã
- å¨åµå¥çç±»æç±»æ¨¡æ¿çå®ä¹ä¸ï¼
- 被ç¨ä½å½åå®ä¾åæåçåµå¥ç±»çåå
- å¨é¨åç¹åæé¨åç¹åçæåçå®ä¹ä¸ï¼
- 类模æ¿çåååéé¨åç¹åçæ¨¡æ¿å®åå表ï¼å ¶ä¸æ¯ä¸ªå®åçä»·äºå ¶å¯¹åºçå½¢å
template <class T> class A { A* p1; // A æ¯å½åå®ä¾å A<T>* p2; // A<T> æ¯å½åå®ä¾å ::A<T>* p4; // ::A<T> æ¯å½åå®ä¾å A<T*> p3; // A<T*> 䏿¯å½åå®ä¾å class B { B* p1; // B æ¯å½åå®ä¾å A<T>::B* p2; // A<T>::B æ¯å½åå®ä¾å typename A<T*>::B* p3; // A<T*>::B 䏿¯å½åå®ä¾å }; }; template <class T> class A<T*> { A<T*>* p1; // A<T*> æ¯å½åå®ä¾å A<T>* p2; // A<T> 䏿¯å½åå®ä¾å }; template <int I> struct B { static const int my_I = I; static const int my_I2 = I+0; static const int my_I3 = my_I; B<my_I>* b3; // B<my_I> æ¯å½åå®ä¾å B<my_I2>* b4; // B<my_I2> 䏿¯å½åå®ä¾å B<my_I3>* b5; // B<my_I3> æ¯å½åå®ä¾å };
注æï¼å¦æåµå¥ç±»æ´¾çäºå ¶å¤å´ç±»æ¨¡æ¿ï¼ååºç±»ä¹å¯ä»¥æ¯å½åå®ä¾åãæ¯å¾ å³ç±»åä½éå½åå®ä¾åçåºç±»ä¸ºå¾ å³åºç±»
template<class T> struct A { typedef int M; struct B { typedef void M; struct C; }; }; template<class T> struct A<T>::B::C : A<T> { M m; // OK, A<T>::M };
符å以䏿 åµçåå被å½ç±»ä¸ºå½åå®ä¾åçæåï¼
- æªéå®åï¼ä¸ºæ é宿¥æ¾å¨å½åå®ä¾åæå ¶éå¾ å³åºç±»ä¸ææ¾å°ã
- éå®åï¼è¥å
¶éå®ç¬¦ï¼
::左侧çååï¼æåå½åå®ä¾åï¼ä¸æ¥æ¾å¨å½åå®ä¾åæå ¶éå¾ å³åºç±»ä¸æ¾å°äºè¯¥åå - ç¨äºç±»æå访é®è¡¨è¾¾å¼çååï¼
x.yæxp->yä¸çyï¼ï¼å ¶ä¸å¯¹è±¡è¡¨è¾¾å¼ï¼xæ*xpï¼æ¯å½åå®ä¾åï¼ä¸æ¥æ¾å¨å½åå®ä¾åæå ¶éå¾ å³åºç±»ä¸æ¾å°äºè¯¥åå
template <class T> class A { static const int i = 5; int n1[i]; // i æä»£å½åå®ä¾åçæå int n2[A::i]; // A::i æä»£å½åå®ä¾åçæå int n3[A<T>::i]; // A<T>::i æä»£å½åå®ä¾åçæå int f(); }; template <class T> int A<T>::f() { return i; // i æä»£å½åå®ä¾åçæå }
å½åå®ä¾åçæåå¯ä¸ºå¾ å³çåéå¾ å³çã
è¥æä¸ªå½åå®ä¾åæåçæ¥æ¾å¨å®ä¾åç¹åå®ä¹ç¹ç»åºä¸åç»æï¼åæ¥æ¾ææ§ä¹ãä½è¦æ³¨æï¼å¨æååè¢«ä½¿ç¨æ¶ï¼å®ä¸ä¼èªå¨ä¼ æ¢æç±»æå访é®è¡¨è¾¾å¼ï¼åªææ¾å¼æå访é®è¡¨è¾¾å¼ææç¤ºå½åå®ä¾åçæåï¼
struct A { int m; }; struct B { int m; };  template<typename T> struct C : A, T { int f() { return this->m; }// 卿¨¡æ¿å®ä¹è¯å¢æ¾å° A::m int g() { return m; } // 卿¨¡æ¿å®ä¹è¯å¢æ¾å° A::m };  template int C<B>::f(); // éè¯¯ï¼æ¾å° A::m å B::m  template int C<B>::g(); // OKï¼å¹¶æªå¨æ¨¡æ¿å®ä¹è¯å¢ä¹ä¸è¿è¡åå访é®è¯æ³ç忢
æªç¥ç¹å
卿¨¡æ¿å®ä¹å ï¼æäºåå被æ¨å¯¼ä¸ºå±äºæä¸ªæªç¥ç¹åï¼ç¹å«æ¯ï¼
- éå®åï¼è¥åºç°äº
::左侧çä»»ä½å忝并éå½åå®ä¾åæåçå¾ å³ç±»å - éå®åï¼å ¶éå®ç¬¦æ¯å½åå®ä¾åï¼ä¸å¨å½åå®ä¾åæä»»ä½å ¶éå¾ å³åºç±»ä¸æ¾ä¸å°è¯¥ååï¼å¹¶åå¨å¾ å³åºç±»
- ç±»æå访é®è¡¨è¾¾å¼ä¸çæååï¼
x.yæxp->yä¸çyï¼ï¼è¥å¯¹è±¡è¡¨è¾¾å¼ï¼xæ*xpï¼çç±»åæ¯å¾ å³ç±»åä¸éå½åå®ä¾å - ç±»æå访é®è¡¨è¾¾å¼ä¸çæååï¼
x.yæxp->yä¸çyï¼ï¼è¥å¯¹è±¡è¡¨è¾¾å¼ï¼xæ*xpï¼çç±»åæ¯å½åå®ä¾åï¼ä¸å¨å½åå®ä¾åæä»»ä½å ¶éå¾ å³åºç±»ä¸æ¾ä¸å°è¯¥ååï¼å¹¶åå¨å¾ å³åºç±»
template<typename T> struct Base {};  template<typename T> struct Derived : Base<T> { void f() { // Derived<T> æä»£å½åå®ä¾å // å½åå®ä¾åæ 'unknown_type' // 使ä¸ä¸ªå¾ å³åºç±»ï¼Base<T>ï¼ // ä»è unknown_type æ¯æªç¥ç¹åçæå typename Derived<T>::unknown_type z; } };  template<> struct Base<int> { // æ¤ç¹åæä¾ä¹ typedef int unknown_type; };
æ¤åç±»å
è®¸å¨æ¨¡æ¿å®ä¹ï¼èéå®ä¾åï¼ç¹æ£æµä¸åé误ï¼
- è¥ä»»ä½æ¨¡æ¿å®ä¹æ¥ææä¸ªéå®åï¼å ¶ä¸çéå®ç¬¦æä»£å½åå®ä¾åï¼ä¸è¯¥å忢éå½åå®ä¾åçæåäº¦éæªç¥ç¹åçæåï¼åç¨åºéè¯æï¼ä¸è¦æ±è¯æï¼ï¼å³ä½¿æ¨¡æ¿å§ç»ä¸è¢«å®ä¾å乿¯å¦æ¤ã
template<class T> class A { typedef int type; void f() { A<T>::type i; // OK ï¼ 'type' æ¯å½åå®ä¾åçæå typename A<T>::other j; // éè¯¯ï¼ // 'other' 䏿¯å½åå®ä¾åçæåï¼ä¸éæªç¥ç¹åçæåï¼ // å 为 A<T>ï¼æåå½åå®ä¾åï¼å¹¶æ æèæ 'other' çå¾ å³åºç±»ã } };
- è¥ä»»ä½æ¨¡æ¿å®ä¹æ¥ææä¸ªæå访é®è¡¨è¾¾å¼çï¼å ¶å¯¹è±¡è¡¨è¾¾å¼æ¯å½åå®ä¾åï¼ä½å ¶å忢éå½åå®ä¾åçæåäº¦éæªç¥ç¹åçæåï¼åç¨åºéè¯æï¼å³ä½¿æ¨¡æ¿å§ç»ä¸è¢«å®ä¾å乿¯å¦æ¤ã
æªç¥ç¹åçæåå§ç»ä¸ºå¾ å³çï¼èä¸åææå¾ å³å䏿 ·ï¼å¨å®ä¾åç¹è¿è¡æ¥æ¾åç»å®ï¼è§ä¸æï¼ã
å¾ å³åç typename æ¶æ§ä¹ç¬¦
卿¨¡æ¿ï¼å æ¬å«å模çï¼ç声ææå®ä¹ä¸ï¼éå½åå®ä¾åæåä¸åå³äºæä¸ªæ¨¡æ¿å½¢åçååï¼ä¸è¢«è®¤ä¸ºæ¯ç±»åï¼é¤é使ç¨å ³é®è¯ typenameï¼æé¤éå ¶å·²è¢«è®¾ç«ä¸ºç±»ååï¼ä¾å¦ç¨ typedef 声ææéè¿ç¨ä½åºç±»åï¼ã
#include <iostream> #include <vector>  int p = 1; template <typename T> void foo(const std::vector<T> &v) {  // std::vector<T>::const_iterator æ¯å¾ å³åï¼ typename std::vector<T>::const_iterator it = v.begin();  // è¥æ 'typename'ï¼åä¸åå 容被解æä¸ºç±»åå¾ å³çæååé // 'const_iterator' åæåé 'p' ç乿³ã // å ä¸ºå¨æ¤å¤æä¸ä¸ªå¯è§çå ¨å± 'p'ï¼æä»¥æ¤æ¨¡æ¿å®ä¹è½ç¼è¯ã std::vector<T>::const_iterator* p;  typedef typename std::vector<T>::const_iterator iter_t; iter_t * p2; // iter_t æ¯å¾ å³åï¼ä½å·²ç¥å®æ¯ç±»åå }  template<typename T> struct S { typedef int value_t; // å½åå®ä¾åçæå void f() { S<T>::value_t n{}; // S<T> å¾ å³ï¼ä½ä¸éè¦ 'typename' std::cout << n << '\n'; } };  int main() { std::vector<int> v; foo(v); // 模æ¿å®ä¾å失败ï¼ç±»å std::vector<int> 䏿 å为 'const_iterator' çæååé S<int>().f(); }
å ³é®è¯ typename ä» å¯ä»¥è¿ç§æ¹å¼ç¨äºéå®åï¼ä¾å¦ T::xï¼ä¹åï¼ä½è¿äºååä¸å¿ å¾ å³ã
对åé typename çæ è¯ç¬¦ä½¿ç¨é常çæéå®å忥æ¾ãè¿ä¸åäºç¨è¯¦è¿°ç±»å说æç¬¦çæ
åµï¼ä¸ç®¡éå®ç¬¦å¦ä½é½ä¸æ¹åæ¥æ¾è§åï¼
struct A { // A æ¥æåµå¥åé X ååµå¥ç±»å struct X struct X {}; int X; }; struct B { struct X { }; // B æ¥æåµå¥ç±»å struct X }; template<class T> void f(T t) { typename T::X x; } void foo() { A a; B b; f(b); // OKï¼å®ä¾å f<B>ï¼T::X æä»£ B::X f(a); // é误ï¼ä¸è½å®ä¾å f<A>ï¼ // å 为 A::X çæéå®ååæ¥æ¾æ¾å°äºæ°æ®æå }
|
å ³é®è¯ typename åªè½å¨æ¨¡æ¿å£°æåå®ä¹ä¸ä½¿ç¨ï¼ä¸åªç¨äºå¯ä»¥ä½¿ç¨å¾ å³åçè¯å¢ä¸ãè¿æé¤äºæ¾å¼ç¹å声æåæ¾å¼å®ä¾å声æã |
(C++11 å) |
|
å ³é®è¯ typename å³ä¾¿å¨æ¨¡æ¿ä¹å¤ä¹å¯ä»¥ä½¿ç¨ã #include <vector>  int main() { typedef typename std::vector<T>::const_iterator iter_t; // C++11 ä¸ OK typename std::vector<int> v; // C++11 ä¸äº¦ OK  } |
(C++11 èµ·) |
|
æäºè¯å¢ä¸ï¼åªæç±»ååè½åæ³å°åºç°ãå¨è¿äºè¯å¢ä¸ï¼åå®å¾ å³çéå®åæåçå°±æ¯ç±»åèä¸å¿ ä½¿ç¨ typenameï¼
|
(C++20 èµ·) |
å¾ å³åç template æ¶æ§ä¹ç¬¦
äºæ¤ç¸ä¼¼ï¼æ¨¡æ¿å®ä¹ä¸ï¼å¹¶éå½åå®ä¾åæåçå¾ å³ååæ ·ä¸è¢«è®¤ä¸ºæ¯æ¨¡æ¿åï¼é¤éä½¿ç¨æ¶æ§ä¹å ³é®è¯ templateï¼æå®å·²è¢«è®¾ç«ä¸ºæ¨¡æ¿åï¼
template<typename T> struct S { template<typename U> void foo(){} };  template<typename T> void bar() { S<T> s; s.foo<T>(); // é误ï¼< 被åæä¸ºå°äºè¿ç®ç¬¦ s.template foo<T>(); // OK }
å ³é®è¯ template ä» å¯ä»¥è¿ç§æ¹å¼ç¨äºè¿ç®ç¬¦ ::ï¼ä½ç¨åè§£æï¼ã->ï¼éè¿æéçæå访é®ï¼å .ï¼æå访é®ï¼ä¹åï¼ä¸åè¡¨è¾¾å¼æ¯ææåæ³ç¤ºä¾ï¼
- T::template foo<X>();
- s.template foo<X>();
- this->template foo<X>();
- typename T::template iterator<int>::value_type v;
ä¸ typename çæ åµä¸æ ·ï¼å³ä½¿ååå¹¶éå¾ å³æå ¶ä½¿ç¨å¹¶æªåºç°äºæ¨¡æ¿çä½ç¨åä¸ (C++11 èµ·)ï¼ä¹å è®¸ä½¿ç¨ template åç¼ã
|
å³ä½¿ template<typename> struct s {}; ::template s<void> q; // å 许ï¼ä½ä¸å¿ é¡» |
(C++17 èµ·) |
æ ¹æ®æ éå®å忥æ¾é对æå访é®è¡¨è¾¾å¼ä¸ç模æ¿åçç¹æ®è§åï¼å½éå¾ å³ç模æ¿å卿å访é®è¡¨è¾¾å¼ä¸åºç°æ¶ï¼-> æ . åï¼ï¼å¦æéè¿è¡¨è¾¾å¼è¯å¢ä¸ç常è§ååæ¥æ¾æ¾å°äºçå ·æç¸åååçç±»æå«å (C++11 èµ·)模æ¿ï¼åä¸éè¦æ¶æ§ä¹ç¬¦ãç¶èï¼è¥è¡¨è¾¾å¼è¯å¢ä¸çæ¥æ¾ææ¾å°ç模æ¿ä¸ç±»è¯å¢ä¸ææ¾å°çä¸åï¼åç¨åºéè¯æã (C++11 å)
template<int> struct A { int value; };  template<class T> void f(T t) { t.A<0>::value; // A çå¸¸è§æ¥æ¾æ¾å°ç±»æ¨¡æ¿ãA<0>::value æåç±» A<0> çæå // t.A < 0; // é误ï¼'<' 被å½å模æ¿å®åå表çèµ·å§ }
ç¼ºé·æ¥å
ä¸åæ´æ¹è¡ä¸ºçç¼ºé·æ¥å追溯å°åºç¨äºä»¥ååºçç C++ æ åã
| DR | åºç¨äº | åºçæ¶çè¡ä¸º | æ£ç¡®è¡ä¸º |
|---|---|---|---|
| CWG 2100 | C++14 | 类模æ¿çéææå常éå°åæªè¢«å为å¼å¾ å³ | å·²åå ¥ |