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++ è¯è¨ > é¨å模æ¿ç¹å
é¨å模æ¿ç¹å
å 许为ç»å®ç±»å«ç模æ¿å®åå®å¶ç±»æ¨¡æ¿æå鿍¡æ¿ (C++14 èµ·)ã
è¯æ³
template < å½¢åå表 > ç±»å
³é®è¯ 类头å < å®åå表 > 声æ
|
|||||||||
template < å½¢åå表 > 声æè¯´æç¬¦åºå 声æç¬¦ < å®åå表 > åå§åå¨(å¯é)
|
(2) | (C++14 èµ·) | |||||||
å ¶ä¸ ç±»å¤´å æ è¯å å声æç类模æ¿ï¼è 声æç¬¦ æ è¯å å声æçå鿍¡æ¿ (C++14 èµ·)ãæ¤å£°æå¿ é¡»ä¸å®æç¹åç主模æ¿å®ä¹å¤äºç¸åçå½å空é´ä¸ï¼æå¯¹äºæå模æ¿ï¼å¤äºç¸åçç±»ä½ç¨åä¸ã
ä¾å¦ï¼
template<class T1, class T2, int I> class A {}; // ä¸»æ¨¡æ¿ Â template<class T, int I> class A<T, T*, I> {}; // #1ï¼é¨åç¹åï¼å ¶ä¸ T2 æ¯æå T1 çæé  template<class T, class T2, int I> class A<T*, T2, I> {}; // #2ï¼é¨åç¹åï¼å ¶ä¸ T1 æ¯æé  template<class T> class A<int, T*, 5> {}; // #3ï¼é¨åç¹åï¼å ¶ä¸ T1 æ¯ intï¼I æ¯ 5ï¼è T2 æ¯æé  template<class X, class T, int I> class A<X, T*, I> {}; // #4ï¼é¨åç¹åï¼å ¶ä¸ T2 æ¯æé
æ ååºä¸çé¨åç¹åçä¾åå å« std::unique_ptrï¼å®å¯¹æ°ç»ç±»åæé¨åç¹åã
å®åå表
ä¸åéå¶åºç¨äºé¨å模æ¿ç¹åç å®åå表ï¼
template<class T1, class T2, int I> class B {}; // ä¸»æ¨¡æ¿ template<class X, class Y, int N> class B<X,Y,N> {}; // é误
|
æ¤å¤ï¼ç¹åå¿ é¡»æ¯ä¸»æ¨¡æ¿æ´ç¹æ® template<int N, typename T1, typename... Ts> struct B; template<typename... Ts> struct B<0, Ts...> { }; // é误ï¼å¹¶ä¸æ´ç¹æ® |
(C++14 èµ·) |
template <int I, int J> struct A {}; template <int I> struct A<I+5, I*2> {}; // é误ï¼I ä¸å¯æ¨å¯¼  template <int I, int J, int K> struct B {}; template <int I> struct B<I, I*2, 2> {}; // OKï¼é¦ä¸ªå½¢å坿¨å¯¼
template <class T, T t> struct C {}; // ä¸»æ¨¡æ¿ template <class T> struct C<T, 1>; // é误ï¼1 çç±»åæ¯ Tï¼ // å®ä¾èµäºå½¢å T  template< int X, int (*array_ptr)[X] > class B {}; // ä¸»æ¨¡æ¿ int array[5]; template< int X > class B<X,&array> { }; // é误ï¼å®åç±»å &array æ¯ int(*)[X]ï¼ä¾èµäºå½¢å X
å忥æ¾
å忥æ¾ä¸ä¼æ¾å°é¨å模æ¿ç¹åãä» å½ä¸»æ¨¡æ¿è¢«ååæ¥æ¾ææ¾å°æ¶ï¼æèèå ¶é¨åç¹åãç¹å«æ¯ä»¤ä¸»æ¨¡æ¿å¯è§ç using 声æï¼äº¦ä»¤é¨åç¹åå¯è§ï¼
namespace N { template<class T1, class T2> class Z { }; // é¨åç¹å } using N::Z; // æä»£ä¸»æ¨¡æ¿ namespace N { template<class T> class Z<T, T*> { }; // é¨åç¹å } Z<int,int*> z; // ååæ¥æ¾æ¾å° N::Zï¼ä¸»æ¨¡æ¿ï¼ï¼ // ç¶å使ç¨å¸¦ T = int çé¨åç¹å
é¨åæåº
å½å®ä¾åç±»æ¨¡æ¿æå鿍¡æ¿ (C++14 èµ·)ï¼ä¸æé¨åç¹åå¯ç¨æ¶ï¼ç¼è¯å¨å¿ é¡»å³å®æ¯ç»§ç»ä½¿ç¨ä¸»æ¨¡æ¿è¿æ¯ä½¿ç¨å ¶é¨åç¹åä¹ä¸ã
// ç»å®ä¸é¢æå®ä¹çæ¨¡æ¿ A A<int, int, 1> a1; // æ å¹é çç¹åï¼ä½¿ç¨ä¸»æ¨¡æ¿ A<int, int*, 1> a2; // ç¨é¨åç¹å #1ï¼(T=int, I=1) A<int, char*, 5> a3; // ç¨é¨åç¹å #3ï¼(T=char) A<int, char*, 1> a4; // ç¨é¨åç¹å #4ï¼(X=int, T=char, I=1) A<int*, int*, 2> a5; // é误ï¼å¹é #2 (T=int, T2=int*, I=2) // å¹é #4 (X=int*, T=int, I=2) // æ ä¸è æ¯å¦ä¸è æ´ç¹æ®
鿣å¼èè¨ï¼âA æ¯ B æ´ç¹æ®âçæææ¯âA æ¥å B ææ¥åç±»åçåéâã
æ£å¼èè¨ï¼ä¸ºå¨é¨åç¹åä¹é´å»ºç«âæ´ç¹æ®âå ³ç³»ï¼é¦å å°å ¶ä¸æ¯ä¸ä¸ªé½è½¬æ¢æä¸åçèè®¾å½æ°æ¨¡æ¿ï¼
- 第ä¸ä¸ªå½æ°æ¨¡æ¿æ¥æä¸ç¬¬ä¸ä¸ªé¨åç¹åç¸åçæ¨¡æ¿å½¢åï¼ä¸åªæä¸ä¸ªå½æ°å½¢åï¼å ¶ç±»åæ¯å¸¦æææ¥èªç¬¬ä¸ä¸ªé¨åç¹åçæ¨¡æ¿å®åç类模æ¿ç¹å
- 第äºä¸ªå½æ°æ¨¡æ¿æ¥æä¸ç¬¬äºä¸ªé¨åç¹åç¸åçæ¨¡æ¿å½¢åï¼ä¸åªæä¸ä¸ªå½æ°å½¢åï¼å ¶ç±»åæ¯å¸¦æææ¥èªç¬¬ä¸ä¸ªé¨åç¹åçæ¨¡æ¿å®åç类模æ¿ç¹åã
ç¶åï¼å¦åä¸ºå½æ°æ¨¡æ¿éè½½æåç䏿 ·ï¼å¯¹å½æ°æ¨¡æ¿æè¡ã
template<int I, int J, class T> struct X { }; // ä¸»æ¨¡æ¿ template<int I, int J> struct X<I, J, int> { static const int s = 1; }; // é¨åç¹å #1 // #1 çèè®¾å½æ°æ¨¡æ¿æ¯ // template<int I, int J> void f(X<I, J, int>); #A  template<int I> struct X<I, I, int> { static const int s = 2; }; // é¨åç¹å #2 // #2 çèè®¾å½æ°æ¨¡æ¿æ¯ // template<int I> void f(X<I, I, int>); #B  int main() { X<2, 2, int> x; // #1 å #2 é½è¦å¹é // 彿°æ¨¡æ¿çé¨åæåºï¼ // #A ä» #Bï¼void(X<I,J,int>) ä» void(X<U1, U1, int>)ï¼æ¨å¯¼ ok // #B ä» #Aï¼void(X<I,I,int>) ä» void(X<U1, U2, int>)ï¼æ¨å¯¼å¤±è´¥ // #B æ´ç¹æ® // #2 为æå®ä¾åçç¹å std::cout << x.s << '\n'; // æå° 2 }
é¨åç¹åçæå
é¨åç¹åçæåçæ¨¡æ¿å½¢ååè¡¨åæ¨¡æ¿å®åå表ï¼å¿ é¡»ä¸é¨åç¹åçå½¢åå表åå®åå表ç¸å¹é ã
æ£å¦ä¸»æ¨¡æ¿çæå䏿 ·ï¼ä» å½å®ä»¬å¨ç¨åºä¸ä½¿ç¨æ¶ï¼æéè¦å®ä¹ã
é¨åç¹åçæåä¸ä¸»æ¨¡æ¿çæåæ å ³ã
é¨åç¹åçæåçæ¾å¼ï¼å ¨ï¼ç¹å以ä¸ä¸»æ¨¡æ¿çæ¾å¼ç¹åç¸åçæ¹å¼å£°æã
template<class T, int I> // ä¸»æ¨¡æ¿ struct A { void f(); // æå声æ };  template<class T, int I> void A<T,I>::f() { } // ä¸»æ¨¡æ¿æåå®ä¹  // é¨åç¹å template<class T> struct A<T,2> { void f(); void g(); void h(); };  // é¨åç¹åçæå template<class T> void A<T,2>::g() { }  // é¨åç¹åçæåçæ¾å¼ï¼å ¨ï¼ç¹å template<> void A<char,2>::h() {}  int main() { A<char,0> a0; A<char,2> a2; a0.f(); // OKï¼ç¨ä¸»æ¨¡æ¿çæåå®ä¹ a2.g(); // OKï¼ç¨é¨åç¹åçæåå®ä¹ a2.h(); // OKï¼ç¨é¨åç¹åçæåçå ¨ç¹åå®ä¹ a2.f(); // é误ï¼é¨åç¹å A<T,2> 䏿 f() çå®ä¹ï¼ä¸ä½¿ç¨ä¸»æ¨¡æ¿ï¼ }
è¥ä¸»æ¨¡æ¿æ¯å¦ä¸ä¸ªç±»æ¨¡æ¿çæåï¼åå ¶é¨åç¹å乿¯å¤å´æ¨¡æ¿ç±»çæåãè¥å¤å´æ¨¡æ¿ç±»è¢«å®ä¾åï¼åæ¯ä¸ªæåé¨åç¹åç声æä¹è¢«å®ä¾åï¼æ¨¡æ¿çææå ¶ä»æåç声æï¼ä½éå ¶å®ä¹ï¼ä¹æ¯ä»¥ç¸åæ¹å¼å®ä¾åçï¼
è¥é对å¤å´ç±»æ¨¡æ¿çç»å®ï¼éå¼ï¼ç¹åè¿è¡ä¸»æå模æ¿çæ¾å¼ï¼å ¨ï¼ç¹åï¼å对å¤å´ç±»æ¨¡æ¿çè¿ä¸ªç¹å忽ç¥è¯¥æå模æ¿çé¨åç¹åã
è¥é对å¤å´ç±»æ¨¡æ¿çç»å®ï¼éå¼ï¼ç¹åè¿è¡æå模æ¿çæä¸ªé¨åç¹åçæ¾å¼ç¹åï¼å对å¤å´ç±»æ¨¡æ¿çè¿ä¸ªç¹åä»ç¶èè主æå模æ¿åå ¶å ¶ä»é¨åç¹åã
template<class T> struct A { // å¤å´ç±»æ¨¡æ¿ template<class T2> struct B {}; // 主æåæ¨¡æ¿ template<class T2> struct B<T2*> {}; // æå模æ¿çé¨åç¹å };  template<> template<class T2> struct A<short>::B {}; // 主æå模æ¿çå ¨ç¹åï¼å¿½ç¥é¨åç¹åï¼ Â A<char>::B<int*> abcip; // 使ç¨é¨åç¹å T2=int A<short>::B<int*> absip; // 使ç¨ä¸»æ¨¡æ¿çå ¨ç¹åï¼å¿½ç¥é¨åç¹åï¼ A<char>::B<int> abci; // 使ç¨ä¸»æ¨¡æ¿
ç¼ºé·æ¥å
ä¸åæ´æ¹è¡ä¸ºçç¼ºé·æ¥å追溯å°åºç¨äºä»¥ååºçç C++ æ åã
| DR | åºç¨äº | åºçæ¶çè¡ä¸º | æ£ç¡®è¡ä¸º |
|---|---|---|---|
| CWG 1315 | C++14 | ä¸è½å¨é¤æ è¯è¡¨è¾¾å¼ä¹å¤çéç±»åå®å表达å¼ä¸ä½¿ç¨æ¨¡æ¿å½¢å | åªè¦è½æ¨å¯¼ï¼è¡¨è¾¾å¼å°± OK |