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 < å½¢åå表 > 彿°å£°æ
|
(1) | ||||||||
template < å½¢åå表 > requires å¶çº¦ 彿°å£°æ
|
(2) | (C++20 èµ·) | |||||||
| 带å ä½ç¬¦å½æ°å£°æ | (3) | (C++20 èµ·) | |||||||
export template < å½¢åå表 > 彿°å£°æ
|
(4) | (C++11 å) | |||||||
è§£é
| å½¢åå表 | - | éç©ºçæ¨¡æ¿å½¢åçéå·åéåè¡¨ï¼æ¯é¡¹æ¯éç±»åå½¢åãç±»åå½¢åãæ¨¡æ¿å½¢åæè¿äºçå½¢åå ä¹ä¸ãä¸ä»»ä½æ¨¡æ¿ä¸æ ·ï¼å½¢åå¯ä»¥æå¶çº¦ã (C++20 èµ·) |
| 彿°å£°æ | - | 彿°å£°æãæå£°æç彿°åæä¸ºæ¨¡æ¿åã |
| å¶çº¦(C++20) | - | å¶çº¦è¡¨è¾¾å¼ï¼å®éå¶æ¤å½æ°æ¨¡æ¿æ¥åçæ¨¡æ¿å½¢åã |
| 带å ä½ç¬¦å½æ°å£°æ(C++20) | - | 彿°å£°æï¼å
¶ä¸è³å°ä¸ä¸ªå½¢åçç±»å使ç¨äºå ä½ç¬¦ auto æ Concept autoï¼æ¨¡æ¿å½¢åå表å°å¯¹æ¯ä¸ªå ä½ç¬¦æ¥æä¸ä¸ªè设形åãï¼è§ä¸æâç®å彿°æ¨¡æ¿âï¼
|
|
|
(C++11 å) |
ç®å彿°æ¨¡æ¿å½å½æ°å£°ææå½æ°æ¨¡æ¿å£°æçå½¢åå表ä¸åºç°å ä½ç¬¦ç±»åï¼ void f1(auto); // ä¸ template<class T> void f(T) ç¸å void f2(C1 auto); // è¥ C1 æ¯æ¦å¿µï¼åä¸ template<C1 T> void f2(T) ç¸å void f3(C2 auto...); // è¥ C2 æ¯æ¦å¿µï¼åä¸ template<C2... Ts> void f3(Ts...) ç¸å void f4(const C3 auto*, C4 auto&); // ä¸ template<C3 T, C4 U> void f4(const T*, U&); ç¸å template <class T, C U> void g(T x, U y, C auto z); // ä¸ template<class T, C U, C W> void g(T x, U y, W z); ç¸å ç®å彿°æ¨¡æ¿å¯ä»¥åææå½æ°æ¨¡æ¿ä¸æ ·è¿è¡ç¹åã template<> void f4<int>(const int*, const double&); // f4<int, const double> çç¹å
|
(C++20 èµ·) |
彿°æ¨¡æ¿å®ä¾å
彿°æ¨¡æ¿èªèº«å¹¶ä¸æ¯ç±»åã彿°æä»»ä½å®ä½ãä»ä» å 嫿¨¡æ¿å®ä¹çæºæä»¶ä¸çæä»»ä½ä»£ç ã为使å¾ä»£ç åºç°å¿ é¡»å®ä¾å模æ¿ï¼å¿ 须确å®å模æ¿å®åï¼ä»¥ä»¤ç¼è¯å¨è½çæä¸ä¸ªå®é ç彿°ï¼æä»ç±»æ¨¡æ¿çæç±»ï¼ã
æ¾å¼å®ä¾å
template è¿åç±»å åå < å®åå表 > ( å½¢åå表 ) ;
|
(1) | ||||||||
template è¿åç±»å åå ( å½¢åå表 ) ;
|
(2) | ||||||||
extern template è¿åç±»å åå < å®åå表 > ( å½¢åå表 ) ;
|
(3) | (C++11 èµ·) | |||||||
extern template è¿åç±»å åå ( å½¢åå表 ) ;
|
(4) | (C++11 èµ·) | |||||||
æ¾å¼å®ä¾åå®ä¹å¼ºå¶å®ä¾åå ¶ææä»£ç彿°ææå彿°ãå®å¯ä»¥åºç°å¨ç¨åºä¸æ¨¡æ¿å®ä¹åçä»»ä½ä½ç½®ï¼è对äºç»å®çå®åå表ï¼åªå 许å®å¨æ´ä¸ªç¨åºä¸åºç°ä¸æ¬¡ã
|
æ¾å¼å®ä¾å声æï¼extern 模æ¿ï¼é»æ¢éå¼å®ä¾åï¼æ¬æ¥ä¼å¯¼è´éå¼å®ä¾åç代ç ï¼å¿ 须使ç¨å·²å¨ç¨åºçå«å¤ææä¾çæ¾å¼å®ä¾åã |
(C++11 èµ·) |
彿°æ¨¡æ¿ç¹åææå彿°æ¨¡æ¿ç¹åçæ¾å¼å®ä¾åä¸ï¼å°¾é¨ç忍¡æ¿å®åï¼è¥è½ä»å½æ°åæ°æ¨å¯¼ï¼åå¯ä»¥ä¸æå®
template<typename T> void f(T s) { std::cout << s << '\n'; }  template void f<double>(double); // å®ä¾å f<double>(double) template void f<>(char); // å®ä¾å f<char>(char)ï¼æ¨å¯¼åºæ¨¡æ¿å®å template void f(int); // å®ä¾å f<int>(int)ï¼æ¨å¯¼åºæ¨¡æ¿å®å
彿°æ¨¡æ¿æç±»æ¨¡æ¿æå彿°çæ¾å¼å®ä¾åä¸è½ä½¿ç¨ inline æ constexprãè¥æ¾å¼å®ä¾åç声ææåäºæä¸ªéå¼å£°æçç¹æ®æå彿°ï¼åç¨åºéè¯æã
æé 彿°çæ¾å¼å®ä¾åä¸è½ä½¿ç¨æ¨¡æ¿å½¢åå表ï¼è¯æ³ (1) ï¼ï¼ä¹å§ç»ä¸éè¦ä½¿ç¨ï¼å ä¸ºè½æ¨å¯¼å®ä»¬ï¼è¯æ³ (2) ï¼ã
æ¾å¼å®ä¾å声æå¹¶ä¸æå¶ inline 彿°ï¼auto 声æï¼å¼ç¨ï¼ä»¥å类模æ¿ç¹åçéå¼å®ä¾åãï¼ä»èï¼å½ä½ä¸ºæ¾å¼å®ä¾å声æç®æ ç inline 彿°è¢« ODR å¼ä½¿ç¨æ¶ï¼å½æ°å°ä¸ºå èèéå¼å®ä¾åï¼ä½æ¤ç¿»è¯åå ä¸ä¸çæå ¶éå è坿¬ï¼
带æé»è®¤å®åç彿°æ¨¡æ¿çæ¾å¼å®ä¾åå®ä¹ï¼ä¸æ¯å¯¹è¯¥å®åç使ç¨ï¼ä¸ä¸ä¼è¯å¾å®ä¾åä¹ï¼
char* p = 0; template<class T> T g(T x = &p) { return x; } template int g<int>(int); // OK å³ä½¿ &p 䏿¯ intã
éå¼å®ä¾å
å½ä»£ç å¨è¦æ±åå¨å½æ°å®ä¹çè¯å¢ä¸ææ¶æä¸ªå½æ°ï¼èè¿ä¸ªç¹å®å½æ°å°æªè¢«æ¾å¼å®ä¾åæ¶ï¼åçéå¼å®ä¾åãè¥æ¨¡æ¿å®åå表è½ä»è¯å¢æ¨å¯¼ï¼åä¸å¿ æä¾å®ã
#include <iostream>  template<typename T> void f(T s) { std::cout << s << '\n'; }  int main() { f<double>(1); // å®ä¾åå¹¶è°ç¨ f<double>(double) f<>('a'); // å®ä¾åå¹¶è°ç¨ f<char>(char) f(7); // å®ä¾åå¹¶è°ç¨ f<int>(int) void (*ptr)(std::string) = f; // å®ä¾å f<string>(string) }
注æï¼å®å
¨çç¥ <> å
许éè½½å³è®®åæ¶æ£éªæ¨¡æ¿ä¸é模æ¿éè½½ã
模æ¿å®åæ¨å¯¼
为å®ä¾åä¸ä¸ªå½æ°æ¨¡æ¿ï¼æ¯ä¸ªæ¨¡æ¿å®åé½å¿ 为已ç¥çï¼ä½å¹¶éå¿ é¡»æå®æ¯ä¸ªæ¨¡æ¿å®åãåªè¦å¯è½ï¼ç¼è¯å¨é½ä¼ä»å½æ°å®åæ¨å¯¼ç¼ºå¤±ç模æ¿å®åãè¿åçäºå°è¯è¿è¡å½æ°è°ç¨æ¶ï¼ä»¥åå彿°æ¨¡æ¿çå°åæ¶ã
template<typename To, typename From> To convert(From f);  void g(double d) { int i = convert<int>(d); // è°ç¨ convert<int,double>(double) char c = convert<char>(d); // è°ç¨ convert<char,double>(double) int(*ptr)(float) = convert; // å®ä¾å convert<int, float>(float) }
æ¤æºå¶ä½¿å¾ä½¿ç¨æ¨¡æ¿è¿ç®ç¬¦å¯è¡ï¼å 为é¤äºå°å ¶éåä¸ºå½æ°è°ç¨è¡¨è¾¾å¼ä¹å¤ï¼ä¸åå¨ä¸ºè¿ç®ç¬¦æå®æ¨¡æ¿å®åçè¯æ³ã
模æ¿å®åæ¨å¯¼å¨å½æ°æ¨¡æ¿çå忥æ¾ï¼å¯è½æ¶åå®åä¾èµæ¥æ¾ï¼ä¹åï¼å¨éè½½å³è®®ä¹é´è¿è¡ã
ç»èè§æ¨¡æ¿å®åæ¨å¯¼ã
æ¾å¼æ¨¡æ¿å®å
彿°æ¨¡æ¿ç模æ¿å®åå¯ä»ä»¥ä¸éå¾è·å¾
- 模æ¿å®åæ¨å¯¼
- é»è®¤æ¨¡æ¿å®å
- æ¾å¼æå®ï¼å¯ä»¥å¨ä¸åè¯å¢ä¸è¿è¡ï¼
- äºå½æ°è°ç¨è¡¨è¾¾å¼ä¸
- å½å彿°å°åæ¶
- å½åå§åå°å½æ°çå¼ç¨æ¶
- 彿ææå彿°æéæ¶
- äºæ¾å¼ç¹åä¸
- äºæ¾å¼å®ä¾åä¸
- äºåå 声æä¸
ä¸åå¨ä¸ºéè½½çè¿ç®ç¬¦ã转æ¢å½æ°åæé 彿°æ¾å¼æå®æ¨¡æ¿å®åçæ¹æ³ï¼å 为å®ä»¬çè°ç¨ä¸å¹¶æªä½¿ç¨å½æ°åã
ææå®ç忍¡æ¿å®åå¿ é¡»ä¸å模æ¿å½¢åå¨ç§ç±»ä¸ç¸å¹é ï¼å³ç±»å对类åï¼éç±»å对éç±»åï¼æ¨¡æ¿å¯¹æ¨¡æ¿ï¼ãä¸è½æå¤äºå½¢åæ°éçå®åï¼é¤éå½¢åä¹ä¸æ¯å½¢åå ï¼è¿ç§æ åµä¸å¯¹æ¯ä¸ªéå å½¢åå¿ é¡»æä¸ä¸ªå®åï¼ã
ææå®çéç±»åå®åå¿ é¡»è¦ä¹ä¸å ¶å¯¹åºçéç±»åæ¨¡æ¿å½¢åçç±»åç¸å¹é ï¼è¦ä¹å¯è½¬æ¢æå®ä»¬ã
ä¸å䏿¨¡æ¿å®åæ¨å¯¼ç彿°å®åï¼ä¾å¦è¥å¯¹åºç模æ¿å®å已被æ¾å¼æå®ï¼ï¼åä¸å°å ¶å¯¹åºå½æ°å½¢åç±»åçéå¼è½¬æ¢ï¼å¦å¨é常éè½½å³è®®ä¸ä¸æ ·ï¼ã
彿é¢å¤çå®åæ¶ï¼æ¨¡æ¿å®åæ¨å¯¼å¯ä»¥å¯¹æ¾å¼æå®ç模æ¿å½¢åå è¿è¡æ©å ï¼
template<class ... Types> void f(Types ... values); void g() { f<int*, float*>(0, 0, 0); // Types = {int*, float*, int} }
模æ¿å®åæ¿æ¢
å½å·²ç»æå®ãæ¨å¯¼åºæä»é»è®¤æ¨¡æ¿å®åè·å¾äºææç模æ¿å®åä¹åï¼å½æ°å½¢åå表ä¸å¯¹æ¨¡æ¿å½¢åçæ¯æ¬¡ä½¿ç¨é½ä¼è¢«æ¿æ¢æå¯¹åºç模æ¿å®åã
彿°æ¨¡æ¿çæ¿æ¢å¤±è´¥ï¼å³ä»¥æ¨å¯¼ææä¾ç模æ¿å®åæ¿æ¢æ¨¡æ¿å½¢å失败ï¼ï¼å°å½æ°æ¨¡æ¿ä»éè½½éä¸ç§»é¤ãè¿å°±å 许使ç¨è®¸å¤æ¹å¼éè¿æ¨¡æ¿å ç¼ç¨æ¥æä½éè½½éï¼ç»èè§ SFINAEã
æ¿æ¢ä¹åï¼æææ°ç»å彿°ç±»åç彿°å½¢åé½è¢«è°æ´ä¸ºæéï¼èææé¡¶å± cv éå®ç¬¦é½ä»å½æ°å½¢åä¸ä¸¢å¼ï¼å¦å¨å¸¸è§å½æ°å£°æä¸ä¸æ ·ï¼ã
å»é¤é¡¶å± cv éå®ç¬¦å¹¶ä¸ä¼å½±åå½¢åå¨å½æ°ä¹å æå±ç°çç±»åï¼
template <class T> void f(T t); template <class X> void g(const X x); template <class Z> void h(Z z, Z* zp);  // 两个ä¸å彿°å ·æç¸åç±»åï¼ä½å¨å½æ°ä¸ï¼t æä¸åç cv éå® f<int>(1); // 彿°ç±»åæ¯ void(int)ï¼t 为 int f<const int>(1); // 彿°ç±»åæ¯ void(int)ï¼t 为 const int  // 两个ä¸å彿°å ·æç¸åç±»ååç¸åç x // ï¼æåè¿ä¸¤ä¸ªå½æ°çæéä¸ç¸çï¼ä¸å½æ°å±é¨çéæåéå¯ä»¥æ¥æä¸åå°åï¼ g<int>(1); // 彿°ç±»åæ¯ void(int) ï¼ x 为 const int g<const int>(1); // 彿°ç±»åæ¯ void(int) ï¼ x 为 const int  // ä» ä¸¢å¼é¡¶å± cv éå®ç¬¦ï¼ h<const int>(1, NULL); // 彿°ç±»åæ¯ void(int, const int*) // z 为 const int ï¼ zp 为 int*
彿°æ¨¡æ¿éè½½
彿°æ¨¡æ¿ä¸é模æ¿å½æ°å¯ä»¥éè½½ã
鿍¡æ¿å½æ°å§ç»ä¸åäºå ·æç¸åç±»åçæ¨¡æ¿ç¹åãä¸å彿°æ¨¡æ¿çç¹åï¼å³ä½¿å ·æç¸åç±»åä¹å§ç»å½¼æ¤ä¸åãä¸¤ä¸ªå ·æç¸åè¿åç±»ååç¸åå½¢åå表ç彿°æ¨¡æ¿æ¯ä¸åçï¼èä¸å¯ç¨æ¾å¼æ¨¡æ¿å®åå表è¿è¡åºåã
å½ä½¿ç¨äºç±»åæéç±»åæ¨¡æ¿å½¢åç表达å¼å¨å½æ°å½¢åå表æè¿åç±»åä¸åºç°æ¶ï¼ä¸ºéè½½çç®çï¼è¯¥è¡¨è¾¾å¼ä¿çä¸ºå½æ°æ¨¡æ¿ç¾åçä¸é¨åï¼
template<int I, int J> A<I+J> f(A<I>, A<J>); // éè½½ #1 template<int K, int L> A<K+L> f(A<K>, A<L>); // å #1 template<int I, int J> A<I-J> f(A<I>, A<J>); // éè½½ #2
对äºä¸¤ä¸ªæ¶å模æ¿å½¢åç表达å¼ï¼è¥ä¸¤ä¸ªå å«è¿äºè¡¨è¾¾å¼ç彿°å®ä¹æ ¹æ® ODR è§åç¸åï¼åç§°å®ä»¬çä»·ï¼å°±æ¯è¯´ï¼é¤äºæ¨¡æ¿å½¢åçå½åå¯ä»¥ä¸åä¹å¤ï¼è¿ä¸¤ä¸ªè¡¨è¾¾å¼å«æç¸åçè®°å·åºåï¼å ¶ä¸çå个ååéè¿å忥æ¾é½è§£æå°ç¸åå®ä½ã两个 lambda 表达å¼å§ç»ä¸çä»·ã (C++20 èµ·)
template <int I, int J> void f(A<I+J>); // 模æ¿éè½½ #1 template <int K, int L> void f(A<K+L>); // çä»·äº #1
|
å¨ç¡®å®ä¸¤ä¸ªå¾ å³è¡¨è¾¾å¼æ¯å¦çä»·æ¶ï¼åªèèå ¶ä¸ææ¶åçåå¾ å³åï¼èä¸èèå忥æ¾çç»æã妿ç¸å模æ¿çå¤ä¸ªå£°æå¨å忥æ¾çç»æä¸ææä¸åï¼å使ç¨é¦ä¸ªè¿ç§å£°æï¼ template <class T> decltype(g(T())) h(); // decltype(g(T())) æ¯å¾ å³ç±»å int g(int); template <class T> decltype(g(T())) h() { // h() çå声æä½¿ç¨è¾æ©çæ¥æ¾ return g(T()); // â¦â¦å°½ç®¡æ¤å¤çæ¥æ¾æ¾å°äº g(int) } int i = h<int>(); // 模æ¿å®åæ¿æ¢å¤±è´¥ï¼å¨ h() çé¦ä¸ªå£°æå¤ g(int) ä¸å¨ä½ç¨åä¸ |
(C++14 èµ·) |
å½è¶³ä¸åæ¡ä»¶æ¶ï¼è®¤ä¸ºä¸¤ä¸ªå½æ°æ¨¡æ¿çä»·
- å®ä»¬å£°æäºåä¸ä½ç¨å
- å®ä»¬å ·æç¸åçåå
- å®ä»¬æ¥æçä»·çæ¨¡æ¿å½¢åå表ï¼å«ä¹æ¯å表é¿åº¦ç¸åï¼ä¸å¯¹äºæ¯å¯¹å¯¹åºçå½¢åï¼ä¸åå å®¹å ¨ä¸ºçï¼
- äºä¸ªå½¢åçç§ç±»ç¸åï¼å为类åãå为éç±»åæå为模æ¿ï¼ï¼
- å®ä»¬é½æ¯å½¢åå æé½ä¸æ¯ï¼
- è¥æ¯éç±»åï¼åå ¶ç±»åçä»·ï¼
- è¥æ¯æ¨¡æ¿ï¼åå ¶æ¨¡æ¿å½¢åçä»·ï¼
|
(C++20 èµ·) |
- å ¶è¿åç±»ååå½¢åå表ä¸ï¼å个æ¶å模æ¿å®åç表达å¼åçä»·
|
(C++20 èµ·) |
对äºä¸¤ä¸ªæ¶å模æ¿å½¢åç表达å¼ï¼å¦æå®ä»¬ä¸çä»·ï¼ä½å¯¹äºä»»ä½ç»å®ç模æ¿å®åéï¼ä¸¤ä¸ªè¡¨è¾¾å¼çæ±å¼é½äº§çç¸åçå¼ï¼åç§°å®ä»¬åè½çä»·ã
å¦æä¸¤ä¸ªå½æ°æ¨¡æ¿æ¬å¯çä»·ï¼ä½å ¶è¿åç±»ååå½¢åå表ä¸ä¸æå¤ä¸ªæ¶å模æ¿å½¢åç表达å¼åè½çä»·ï¼åç§°å®ä»¬åè½çä»·ã
|
å¦å¤ï¼å¦æä¸ºä¸¤ä¸ªå½æ°æ¨¡æ¿æå®çå¶çº¦ä¸åï¼ä½å®ä»¬æ¥åä¸è¢«ç¸åçæ¨¡æ¿å®åå表çéåææ»¡è¶³ï¼åå®ä»¬åè½çä»·ä½ä¸çä»·ã |
(C++20 èµ·) |
è¥ç¨åºå«æåè½çä»·ä½ä¸çä»·ç彿°æ¨¡æ¿å£°æï¼åç¨åºéè¯æï¼ä¸è¦æ±è¯æã
// çä»· template <int I> void f(A<I>, A<I+10>); // éè½½ #1 template <int I> void f(A<I>, A<I+10>); // éè½½ #1 çå声æ  // ä¸çä»· template <int I> void f(A<I>, A<I+10>); // éè½½ #1 template <int I> void f(A<I>, A<I+11>); // éè½½ #2  // åè½çä»·ä½ä¸çä»· // ç¨åºéè¯æï¼ä¸è¦æ±è¯æ template <int I> void f(A<I>, A<I+10>); // éè½½ #1 template <int I> void f(A<I>, A<I+1+2+3+4>); // åè½çä»·
å½åä¸ä¸ªå½æ°æ¨¡æ¿ç¹åä¸å¤äºä¸ä¸ªéè½½ç彿°æ¨¡æ¿ç¸å¹é æ¶ï¼è¿éå¸¸ç±æ¨¡æ¿å®åæ¨å¯¼æå¯¼è´ï¼ï¼è¿è¡éè½½å½æ°æ¨¡æ¿çé¨åæåºä»¥éæ©æä½³å¹é ã
å ·ä½èè¨ï¼å¨ä»¥ä¸æ å½¢ä¸åçé¨åæåºï¼
template<class X> void f(X a); template<class X> void f(X* a); int* p; f(p);
template<class X> void f(X a); template<class X> void f(X* a); void (*p)(int*) = &f;
operator delete 以å¹é
å¸ç½® operator new æ¶
| æ¬èæªå®æ åå ï¼å°ç¤ºä¾ |
template<class X> void f(X a); // 第ä¸ä¸ª template f template<class X> void f(X* a); // 第äºä¸ª template f template<> void f<>(int *a) {} // æ¾å¼ç¹å // 模æ¿å®åæ¨å¯¼åºç°ä¸¤ä¸ªåéï¼ // foo<int*>(int*) ä¸ f<int>(int*) // é¨åæåºéæ© f<int>(int*)ï¼å ä¸ºå®æ´ç¹æ®
鿣å¼èè¨ï¼âA æ¯ B æ´ç¹æ®âæå³çâA æ¯ B æ¥åæ´å°çç±»åâã
æ£å¼èè¨ï¼ä¸ºç¡®å®ä»»æä¸¤ä¸ªå½æ°æ¨¡æ¿ä¸åªä¸ªæ´ç¹æ®ï¼é¨åæåºè¿ç¨é¦å 对两个模æ¿ä¹ä¸è¿è¡ä»¥ä¸åæ¢ï¼
- å¯¹äºæ¯ä¸ªç±»åãéç±»å忍¡æ¿å½¢åï¼å æ¬å½¢åå ï¼çæä¸ä¸ªå¯ä¸çèæç±»åãå¼ææ¨¡æ¿ï¼ä»¥ä¹å¨æ¨¡æ¿ç彿°ç±»åä¸è¿è¡æ¿æ¢
- è¥è¦æ¯è¾çä¸¤ä¸ªå½æ°æ¨¡æ¿ä¸åªæä¸ä¸ªæ¯æå彿°ï¼ä¸è¯¥å½æ°æ¨¡æ¿æ¯æä¸ªç±»
Açééææåï¼ååå ¶å½¢åå表æå ¥ä¸ä¸ªæ°çé¦ä¸ªå½¢åï¼å½æå彿°æ¨¡æ¿ä¸º && é宿¶ï¼å ¶ç±»å为cv A&&ï¼å¦å为cv A&ï¼å ¶ä¸ cv æ¯æå彿°æ¨¡æ¿ç cv éå®ï¼ââè¿æå©äºå¯¹è¿ç®ç¬¦çæåºï¼å®ä»¬æ¯åæ¶ä½ä¸ºæåä¸éæå彿°æ¥æ¾çï¼
struct A {}; template<class T> struct B { template<class R> int operator*(R&); // #1 }; template<class T, class R> int operator*(T&, R&); // #2 int main() { A a; B<A> b; b * a; // 模æ¿å®åæ¨å¯¼å¯¹äº int B<A>::operator*(R&) ç»åº R=A // å¯¹äº int operator*(T&, R&)ï¼T=B<A>ï¼R=A // 为è¿è¡é¨åæåºï¼æå template B<A>::operator* // è¢«åæ¢æ template<class R> int operator*(B<A>&, R&); // int operator*( T&, R&) T=B<A>, R=A // ä¸ int operator*(B<A>&, R&) R=A é´çé¨åæåº // éæ© int operator*(B<A>&, A&) 为æ´ç¹æ®è
卿䏿¹æè¿°åæ¢ä¸¤ä¸ªæ¨¡æ¿ä¹ä¸åï¼ä»¥åæ¢å模æ¿ä¸ºå®å模æ¿ï¼ä»¥å¦ä¸æ¨¡æ¿ç忍¡æ¿ç±»åä¸ºå½¢åæ¨¡æ¿ï¼æ§è¡æ¨¡æ¿å®åæ¨å¯¼ãç¶å以第äºä¸ªæ¨¡æ¿ï¼è¿è¡åæ¢åï¼ä¸ºå®åï¼ä»¥ç¬¬ä¸ä¸ªæ¨¡æ¿çåå§å½¢å¼ä¸ºå½¢åéå¤è¿ä¸è¿ç¨ã
ç¨äºç¡®å®é¡ºåºçç±»ååå³äºè¯å¢ï¼
- å¨å½æ°è°ç¨çè¯å¢ä¸ï¼è¿äºç±»åæ¯å¨è¿ä¸ªå½æ°è°ç¨ä¸å ·æå®åç彿°å½¢åçç±»åï¼ä¸èèé»è®¤å½æ°å®åãå½¢åå åçç¥å·å½¢åââè§ä¸æï¼
- å¨è°ç¨ç¨æ·å®ä¹ç转æ¢å½æ°çè¯å¢ä¸ï¼ä½¿ç¨è½¬æ¢å½æ°æ¨¡æ¿çè¿åç±»å
- å¨å ¶ä»è¯å¢ä¸ï¼ä½¿ç¨å½æ°æ¨¡æ¿ç±»å
形忍¡æ¿ä¸çæ¯ä¸ªä»¥ä¸ååºçç±»åé½è¢«æ¨å¯¼ãæ¨å¯¼å¼å§åï¼ä»¥ä¸åæ¹å¼å¯¹å½¢å模æ¿çæ¯ä¸ªå½¢å P åå®å模æ¿ç对åºå®å A è¿è¡è°æ´ï¼
- è¥
Pä¸Aæ¤åå为å¼ç¨ç±»åï¼åç¡®å®åªä¸ªæ´ä¸º cv éå®ï¼å ¶ä»æææ åµä¸ï¼å°±é¨åæåºèè¨é½å¿½ç¥ cv éå®ï¼ - è¥
Pæ¯å¼ç¨ç±»åï¼åä»¥å ¶æå¼ç¨çç±»åæ¿æ¢å® - è¥
Aæ¯å¼ç¨ç±»åï¼åä»¥å ¶æå¼ç¨çç±»åæ¿æ¢å® - è¥
Pæ cv éå®ï¼åPè¢«æ¿æ¢ä¸ºèªèº«çæ cv éå®çæ¬ - è¥
Aæ cv éå®ï¼åAè¢«æ¿æ¢ä¸ºèªèº«çæ cv éå®çæ¬
å¨è¿äºè°æ´åï¼éµå¾ªä»ç±»åè¿è¡æ¨¡æ¿å®åæ¨å¯¼è§åï¼ä» A æ¨å¯¼ P ã
è¥ P æ¯å½æ°å½¢åå ï¼åå®å模æ¿çæ¯ä¸ªå©ä½å½¢åç±»åçç±»å Aï¼é½ä¸è¯¥å½æ°åæ°å ç声æç¬¦æ è¯çç±»å P è¿è¡æ¯è¾ãæ¯æ¬¡æ¯è¾é½ä¸ºè¯¥å½æ°åæ°å æå±å¼ç模æ¿åæ°å ä¸çåç»§ä½ç½®çè¿è¡æ¨¡æ¿å®åçæ¨å¯¼ã
è¥ A ä»å½æ°åæ°å åæ¢èæ¥ï¼åæ¨å¯¼å¤±è´¥ã (C++14 å)å°å®ä¸å½¢å模æ¿çæ¯ä¸ªå©ä½å½¢åç±»åè¿è¡æ¯è¾ã (C++14 èµ·)
è¥åæ¢åçæ¨¡æ¿ 1 çå®å A å¯ç¨æ¥æ¨å¯¼æ¨¡æ¿ 2 ç对åºå½¢å Pï¼ä½åä¹ä¸å¯ï¼å对äºä»è¿ä¸å¯¹ P/A ææ¨å¯¼çç±»åèè¨ï¼è¿ä¸ª A æ¯ P æ´ç¹æ®ã
è¥å忍坼åæåï¼ä¸å P ä¸ A æ¯å¼ç¨ç±»åï¼ååéå çæµè¯ï¼
- è¥
Aæ¯å·¦å¼å¼ç¨èPæ¯å³å¼å¼ç¨ï¼å认为 A æ¯ P æ´ç¹æ® - è¥
Aæ¯Pæ´æ cv éå®ï¼å认为 A æ¯ P æ´ç¹æ®
ææå ¶ä»æ åµä¸ï¼å¯¹äºè¿ä¸å¯¹ P/A ææ¨å¯¼çç±»åèè¨ï¼æ²¡ææ¨¡æ¿æ¯å¦ä¸ä¸ªæ´ç¹æ®ã
å¨ä»¥ä¸¤ä¸ªæ¹åèèæ¯ä¸ª P ä¸ A åï¼è¥å¯¹äºæèèçæ¯ä¸ªç±»åï¼
- æ¨¡æ¿ 1 对ææç±»åè³å°ä¸æ¨¡æ¿ 2 䏿 ·ç¹æ®
- æ¨¡æ¿ 1 对æäºç±»åæ¯æ¨¡æ¿ 2 ç¹æ®
- æ¨¡æ¿ 2 对任ä½ç±»åé½ä¸æ¯æ¨¡æ¿ 1 æ´ç¹æ®ï¼æå¹¶é对任ä½ç±»åé½è³å°ä¸æ ·ç¹æ®
忍¡æ¿ 1 æ¯æ¨¡æ¿ 2 æ´ç¹æ®ãè¥ä¸è¿°æ¡ä»¶å¨åæ¢æ¨¡æ¿é¡ºåºå为çï¼åæ¨¡æ¿ 2 æ¯æ¨¡æ¿ 1 æ´ç¹æ®ãå¦åï¼æ²¡ææ¨¡æ¿æ¯å¦ä¸ä¸ªæ´ç¹æ®ã æå¹³çæ åµä¸ï¼è¥ä¸ä¸ªå½æ°æ¨¡æ¿æä¸ä¸ªå°¾é¨çå½¢åå èå¦ä¸ä¸ªæ²¡æï¼å认为带æè¢«å¿½ç¥çå½¢åè æ¯æç©ºå½¢åå è æ´ç¹æ®ã
è¥å¨èèææçé载模æ¿å¯¹ä¹åï¼æä¸ä¸ªæ æ§ä¹å°æ¯ææå ¶ä»ç齿´ç¹æ®ï¼åéæ©è¿ä¸ªæ¨¡æ¿çç¹åï¼å¦åç¼è¯å¤±è´¥ã
å¨ä¸å示ä¾ä¸ï¼èæå®å被称为 U1, U2
template<class T> void f(T); // æ¨¡æ¿ #1 template<class T> void f(T*); // æ¨¡æ¿ #2 template<class T> void f(const T*); // æ¨¡æ¿ #3 void m() { const int* p; f(p); // éè½½å³è®®éåï¼ #1ï¼void f(T ) [T = const int *] // ã #2ï¼void f(T*) [T = const int] // ã #3ï¼void f(const T *) [T = int] // é¨åæåº // #1 ä»åæ¢åç #2ï¼void(T) ä» void(U1*)ï¼P=T A=U1*ï¼æ¨å¯¼ okï¼T=U1* // #2 ä»åæ¢åç #1ï¼void(T*) ä» void(U1)ï¼P=T* A=U1ï¼æ¨å¯¼å¤±è´¥ // å¯¹äº T èè¨ #2 æ¯ #1 æ´ç¹æ® // #1 ä»åæ¢åç #3ï¼void(T) ä» void(const U1*)ï¼P=T, A=const U1*ï¼ok // #3 ä»åæ¢åç #1ï¼void(const T*) ä» void(U1)ï¼P=const T*, A=U1ï¼å¤±è´¥ // å¯¹äº T èè¨ #3 æ¯ #1 æ´ç¹æ® // #2 ä»åæ¢åç #3ï¼void(T*) ä» void(const U1*)ï¼P=T* A=const U1*ï¼ok // #3 ä»åæ¢åç #2ï¼void(const T*) ä» void(U1*)ï¼P=const T* A=U1*ï¼å¤±è´¥ // å¯¹äº T èè¨ #3 æ¯ #2 æ´ç¹æ® // ç»æï¼#3 è¢«éæ© // æ¢è¨ä¹ï¼f(const T*) æ¯ f(T) æ f(T*) æ´ç¹æ® }
template<class T> void f(T, T*); // #1 template<class T> void f(T, int*); // #2 void m(int* p) { f(0, p); // #1 çæ¨å¯¼ï¼void f(T, T*) [T = int] // #2 çæ¨å¯¼ï¼void f(T, int*) [T = int] // é¨åæåºï¼ // #1 ä» #2ï¼void(T,T*) ä» void(U1,int*)ï¼P1=T, A1=U1ï¼T=U1 // P2=T*, A2=int*ï¼T=intï¼å¤±è´¥ // #2 ä» #1ï¼void(T,int*) ä» void(U1, U2*)ï¼P1=T A1=U1ï¼T=U1 // P2=int* A2=U2*ï¼å¤±è´¥ // å¯¹äº T èè¨æ 䏿´ç¹æ®ï¼è°ç¨ææ§ä¹ }
template<class T> void g(T); // æ¨¡æ¿ #1 template<class T> void g(T&); // æ¨¡æ¿ #2 void m() { float x; g(x); // ä» #1 æ¨å¯¼ï¼void g(T ) [T = float] // ä» #2 æ¨å¯¼ï¼void g(T&) [T = float] // é¨åæåº // #1 ä» #2ï¼void(T) ä» void(U1&)ï¼P=T, A=U1ï¼è°æ´åï¼ï¼ok // #2 ä» #1ï¼void(T&) ä» void(U1)ï¼P=Tï¼è°æ´åï¼ A=U1 ï¼ok // å¯¹äº T èè¨æ 䏿´ç¹æ®ï¼è°ç¨ææ§ä¹ }
template<class T> struct A { A(); };  template<class T> void h(const T&); // #1 template<class T> void h(A<T>&); // #2 void m() { A<int> z; h(z); // ä» #1 æ¨å¯¼ï¼void h(const T &) [T = A<int>] // ä» #2 æ¨å¯¼ï¼void h(A<T> &) [T = int] // é¨åæåº // #1 ä» #2ï¼void(const T&) ä» void(A<U1>&)ï¼P=T A=A<U1>ï¼ok T=A<U1> // #2 ä» #1ï¼void(A<T>&) ä» void(const U1&)ï¼P=A<T> A=const U1ï¼å¤±è´¥ // å¯¹äº T èè¨ #2 æ¯ #1 æ´ç¹æ®  const A<int> z2; h(z2); // ä» #1 æ¨å¯¼ï¼void h(const T&) [T = A<int>] // ä» #2 æ¨å¯¼ï¼void h(A<T>&) [T = int]ï¼ä½æ¿æ¢å¤±è´¥ // åªæä¸ä¸ªå¯éæ©çéè½½ï¼ä¸å°è¯é¨åæåºï¼è°ç¨ #1 }
å 为å¨è°ç¨è¯å¢ä¸åªèèææç¡®çè°ç¨å®åçå½¢åï¼æ 没ææç¡®çè°ç¨å®åçå½¢åï¼å æ¬å½æ°å½¢åå ãçç¥å·å½¢ååæé»è®¤å®åçå½¢åå被忽ç¥ï¼
template<class T> void f(T); // #1 template<class T> void f(T*, int=1); // #2 void m(int* ip) { int* ip; f(ip); // è°ç¨ #2ï¼T* æ¯ T æ´ç¹æ®ï¼ }
template<class T> void g(T); // #1 template<class T> void g(T*, ...); // #2 void m(int* ip) { g(ip); // è°ç¨ #2ï¼T* æ¯ T æ´ç¹æ®ï¼ }
template<class T, class U> struct A { }; template<class T, class U> void f(U, A<U,T>* p = 0); // #1 template< class U> void f(U, A<U,U>* p = 0); // #2 void h() { f<int>(42, (A<int, int>*)0); // è°ç¨ #2 f<int>(42); // éè¯¯ï¼æ§ä¹ }
template<class T > void g(T, T = T()); // #1 template<class T, class... U> void g(T, U ...); // #2 void h() { g(42); // éè¯¯ï¼æ§ä¹ }
template<class T, class... U> void f(T, U...); // #1 template<class T > void f(T); // #2 void h(int i) { f(&i); // è°ç¨ #2 å 为形åå 䏿 å½¢åä¹é´çå³èè§å // ï¼æ³¨æï¼å¨ DR692 ä¸ DR1395 ä¹é´æ¶ææ§ä¹ï¼ }
template<class T, class... U> void g(T*, U...); // #1 template<class T > void g(T); // #2 void h(int i) { g(&i); // OKï¼è°ç¨ #1ï¼T* æ¯ T æ´ç¹æ®ï¼ }
template <class ...T> int f(T*...); // #1 template <class T> int f(const T&); // #2 f((int*)0); // OKï¼éæ© #1 // ï¼DR1395 ä¹åææ§ä¹ï¼å 为两个æ¹åçæ¨å¯¼å失败ï¼
template<class... Args> void f(Args... args); // #1 template<class T1, class... Args> void f(T1 a1, Args... args); // #2 template<class T1, class T2> void f(T1 a1, T2 a2); // #3 f(); // è°ç¨ #1 f(1, 2, 3); // è°ç¨ #2 f(1, 2); // è°ç¨ #3ï¼éå忍¡æ¿ #3 æ¯å忍¡æ¿ #1 ä¸ #2 æ´ç¹æ®
å¨é¨åæåºè¿ç¨ç模æ¿å®åæ¨å¯¼æé´ï¼è¥å®åæªè¢«é¨åæåºæèèçä»»ä½ç±»å使ç¨ï¼åä¸è¦æ±è¯¥å®å䏿¨¡æ¿å½¢åç¸å¹é
template <class T> T f(int); // #1 template <class T, class U> T f(U); // #2 void g() { f<int>(1); // #1 çç¹å为æ¾å¼ï¼T f(int) [T = int] // #2 çç¹å为æ¨å¯¼ï¼T f(U) [T = int, U = int] // é¨åæåºï¼ä» èèå®åç±»åï¼ // #1 ä» #2ï¼T(int) ä» U1(U2)ï¼å¤±è´¥ // #2 ä» #1ï¼T(U) ä» U1(int)ï¼okï¼U=int, T æªä½¿ç¨ // è°ç¨ #1 }
对å 嫿¨¡æ¿å½¢åå ç彿°æ¨¡æ¿è¿è¡çé¨åæåºï¼ä¸ä¸ºè¿äºæ¨¡æ¿å½¢åå ææ¨å¯¼çå®åæ°éæ å ³ã
template<class...> struct Tuple { }; template< class... Types> void g(Tuple<Types ...>); // #1 template<class T1, class... Types> void g(Tuple<T1, Types ...>); // #2 template<class T1, class... Types> void g(Tuple<T1, Types& ...>); // #3  g(Tuple<>()); // è°ç¨ #1 g(Tuple<int, float>()); // è°ç¨ #2 g(Tuple<int, float&>()); // è°ç¨ #3 g(Tuple<int>()); // è°ç¨ #3
| æ¬èæªå®æ åå ï¼14.8.3[temp.over] |
为ç¼è¯å¯¹å½æ°æ¨¡æ¿çè°ç¨ï¼ç¼è¯å¨å¿ é¡»å¨é模æ¿éè½½ãæ¨¡æ¿éè½½åæ¨¡æ¿éè½½çç¹åé´ä½åºå³å®ã
template< class T > void f(T); // #1ï¼æ¨¡æ¿éè½½ template< class T > void f(T*); // #2ï¼æ¨¡æ¿éè½½ void f(double); // #3ï¼é模æ¿éè½½ template<> void f(int); // #4ï¼#1 çç¹å  f('a'); // è°ç¨ #1 f(new int(1)); // è°ç¨ #2 f(1.0); // è°ç¨ #3 f(1); // è°ç¨ #4
彿°éè½½ vs 彿°ç¹å
注æï¼åªæé模æ¿å主模æ¿éè½½åä¸éè½½å³è®®ãç¹å并䏿¯éè½½èä¸åèèãåªæå¨éè½½å³è®®éæ©æä½³å¹é ç䏻彿°æ¨¡æ¿åï¼ææ£éªå ¶ç¹å以æ¥çä½è 为æä½³å¹é ã
template< class T > void f(T); // #1ï¼ææç±»åçéè½½ template<> void f(int*); // #2ï¼é对æå int çæéç #1 çç¹å template< class T > void f(T*); // #3ï¼æææéç±»åçéè½½  f(new int(1)); // è°ç¨ #3ï¼è½ç¶ #1 çç¹åæ¯å®ç¾å¹é
å¨å¯¹ç¿»è¯åå ç头æä»¶è¿è¡æåºæ¶ï¼è®°ä½æ¤è§åå¾éè¦ãæå ³å½æ°éè½½ä¸å½æ°ç¹åä¹é´çæ´å¤ç¤ºä¾ï¼å±å¼äºä¸ï¼
|
|
|
é¦å èèä¸äºä¸ä½¿ç¨å®åä¾èµæ¥æ¾çåºæ¯ã对äºè¿ç§æ åµï¼æä»¬ä½¿ç¨è°ç¨ (f)(t)ãå¦ ADL ä¸çæè¿°ï¼å°å½æ°åå 卿¬å·ä¸å¯æå¶å®åä¾èµæ¥æ¾ã
è¿è¡æ¤ä»£ç #include <iostream> struct A{}; template<class T> void f(T) {std::cout << "#1\n";} // éè½½ #1 å¨ f() POR å template<class T> void f(T*) {std::cout << "#2\n";} // éè½½ #2 å¨ f() POR å template<class T> void g(T* t) { (f)(t); // f() POR }  int main() { A *p=nullptr; g(p); } // POI of g() and f()  // #1 ä¸ #2 é½è¢«æ·»å å°åéåè¡¨ï¼ // éæ© #2 å ä¸ºå®æ¯è¾å¥½çå¹é ã è¾åºï¼ #2
è¿è¡æ¤ä»£ç #include <iostream> struct A{}; template<class T> void f(T) {std::cout << "#1\n";} // #1 template<class T> void g(T* t) { (f)(t); // f() POR } template<class T> void f(T*) {std::cout << "#2\n";} // #2  int main() { A *p=nullptr; g(p); } // POI of g() and f()  // ä» æ·»å #1 å°åéå表ï¼#2 å®ä¹äº POR ä¹åï¼ // å æ¤ï¼å³ä½¿å®æ¯è¾ä½³å¹é ï¼å®ä¹ä¸ä¸ºéè½½æèèã è¾åºï¼ #1
è¿è¡æ¤ä»£ç #include <iostream> struct A{}; template<class T> void f(T) {std::cout << "#1\n";} // #1 template<class T> void g(T* t) { (f)(t); // f() POR } template<> void f<>(A*) {std::cout << "#3\n";} // #3  int main() { A *p=nullptr; g(p); } // g() ä¸ f() ç POI  // æ·»å #1 å°åéå表ï¼#3 æ¯å®ä¹äº POR åçè¾å¥½å¹é ãåéåè¡¨ç±æç»è¢«éæ©ç #1 ç»æã // ä¹åï¼å£°æäº POI åç #1 çæ¾å¼ç¹å #3 è¢«éæ©ï¼å ä¸ºå®æ¯è¾å¥½çå¹é ã // æ¤è¡ä¸ºç± 14.7.3/6 [temp.expl.spec] ææ§ä¸ä¸ ADL æ å ³ã è¾åºï¼ #3
è¿è¡æ¤ä»£ç #include <iostream> struct A{}; template<class T> void f(T) {std::cout << "#1\n";} // #1 template<class T> void g(T* t) { (f)(t); // f() POR } template<class T> void f(T*) {std::cout << "#2\n";} // #2 template<> void f<>(A*) {std::cout << "#3\n";} // #3  int main() { A *p=nullptr; g(p); } // g() ä¸ f() ç POI  // #1 æ¯åéå表çå¯ä¸æåä¸å®è¢«æç»éæ©ã // ä¹åï¼è·³è¿æ¾å¼ç¹å #3ï¼å 为å®å®é ç¹åäºå£°æäº POR åç #2ã è¾åºï¼ #1
è¿è¡æ¤ä»£ç #include <iostream> struct A{}; template<class T> void f(T) {std::cout << "#1\n";} // #1 template<class T> void g(T* t) { f(t); // f() POR } template<class T> void f(T*) {std::cout << "#2\n";} // #2  int main() { A *p=nullptr; g(p); } // g() ä¸ f() ç POI  // #1 被ä½ä¸ºå¸¸è§æ¥æ¾çç»ææ·»å å°åéåè¡¨ï¼ // #2 å®ä¹äº POR ä¹åä½ç»ç± ADL æ¥æ¾æ·»å å°åéå表ã // #2 ä½ä¸ºè¾å¥½çå¹é è¢«éæ©ã è¾åºï¼ #2
è¿è¡æ¤ä»£ç #include <iostream> struct A{}; template<class T> void f(T) {std::cout << "#1\n";} // #1 template<class T> void g(T* t) { f(t); // f() POR } template<> void f<>(A*) {std::cout << "#3\n";} // #3 template<class T> void f(T*) {std::cout << "#2\n";} // #2  int main() { A *p=nullptr; g(p); } // g() ä¸ f() ç POI  // #1 被ä½ä¸ºå¸¸è§æ¥æ¾çç»ææ·»å å°åéåè¡¨ï¼ // #2 å®ä¹äº POR ä¹åä½ç»ç± ADL æ¥æ¾æ·»å å°åéå表ã // ä½ä¸ºè¾å¥½çå¹é ï¼ä»å主模æ¿ä¸éæ© #2ã // å 为 #3 声æäº #2 ä¹åï¼æ 宿¯ #1 çæ¾å¼ç¹åã // ä»èæç»éæ©ä¸º #2ã è¾åºï¼ #2
è¿è¡æ¤ä»£ç #include <iostream> struct A{}; template<class T> void f(T) {std::cout << "#1\n";} // #1 template<class T> void g(T* t) { f(t); // f() POR } template<class T> void f(T*) {std::cout << "#2\n";} // #2 template<> void f<>(A*) {std::cout << "#3\n";} // #3  int main() { A *p=nullptr; g(p); } // g() ä¸ f() ç POI  // #1 被ä½ä¸ºå¸¸è§æ¥æ¾çç»ææ·»å å°åéåè¡¨ï¼ // #2 å®ä¹äº POR ä¹åä½ç»ç± ADL æ¥æ¾æ·»å å°åéå表ã // ä½ä¸ºè¾å¥½çå¹é ï¼ä»å主模æ¿ä¸éæ© #2ã // å 为 #3 声æäº #2 ä¹åï¼æ 宿¯ #2 çæ¾å¼ç¹åï¼ // ä»è被é为è°ç¨ç彿°ã è¾åºï¼ #3
|
æå ³éè½½å³è®®ç详ç»è§åï¼è§éè½½å³è®®ã
彿°æ¨¡æ¿ç¹å
| æ¬èæªå®æ åå ï¼14.8[temp.fct.spec] (note that 14.8.1[temp.arg.explicit] å·²å¨å ¨ç¹åä¸é¢ï¼è¦ä¹è®©å½æ°ç¹åå°è¿éï¼å¤±å»é¨åç¹åãä¸å½æ°éè½½ç交äºï¼æä» å¼ç¨é£è¾¹ |
ç¼ºé·æ¥å
ä¸åæ´æ¹è¡ä¸ºçç¼ºé·æ¥å追溯å°åºç¨äºä»¥ååºçç C++ æ åã
| DR | åºç¨äº | åºçæ¶çè¡ä¸º | æ£ç¡®è¡ä¸º |
|---|---|---|---|
| CWG 1395 | C++14 | ä»å½¢åå æ¨å¯¼ A æ¶å¤±è´¥ï¼ä¸å¯¹äºç©ºå½¢åå æ²¡æå³èè§å | å 许æ¨å¯¼ï¼æ·»å å³èè§å |