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++ è¯è¨ > äºå¡æ§å å
äºå¡æ§å å
äºå¡æ§å åï¼transactional memoryï¼æ¯å¨äºå¡ä¸ç»åè¯å¥ç»çå¹¶å忥æºå¶ï¼äºå¡å ·æ
- ååæ§ï¼atomicï¼ï¼è¦ä¹è¯å¥å ¨é¨åçï¼è¦ä¹å ¨é¨ä¸åçï¼
- é离æ§ï¼isolatedï¼ï¼äºå¡ä¸çè¯å¥ä¸ä¼è§å¯å°å¦ä¸äºå¡åå ¥ä¸åï¼å³ä½¿å®ä»¬å¹¶è¡æ§è¡ï¼
å ¸åå®ç°å¨åæ¯æçå¹³å°ä¸å°ç¡¬ä»¶äºå¡æ§å å使ç¨å°æè´ï¼ä¾å¦ï¼ç´è³åæ´é饱åï¼ï¼åéåå°è½¯ä»¶äºå¡æ§å åï¼åè 常以ä¹è§å¹¶åï¼optimistic concurrencyï¼æ¥å®ç°ï¼è¥å¦ä¸äºå¡æ´æ°äºäºå¡æç¨çæäºåéï¼å该äºå¡ä¼å®éå°éè¯ãç±äºè¿ä¸ªåå ï¼å¯éè¯äºå¡ï¼âåååâï¼åªè½è°ç¨äºå¡å®å ¨ç彿°ã
注æï¼å¨äºå¡å åäºå¡å¤è®¿é®åéèæ å ¶ä»çå¤é¨åæ¥ï¼æ¯æ°æ®ç«äºã
è¥æ¯æåè½ç¹æ§æµè¯ï¼åæ¬é¡µææè¿°çåè½ç¹æ§ï¼ç±å ·æå¤§äºæçäº 201505 çå¼çå®å¸¸é __cpp_transactional_memory æ æã
忥å
synchronized å¤åè¯å¥
å¦åå¨ä¸ä¸ªå ¨å±é䏿§è¡å¤åè¯å¥ï¼ç¨åºä¸çæææå¤å±åæ¥åé½ä»¥ä¸ä¸ªåç¬çå ¨åºæ§è¡ãå¨è¯¥é¡ºåºä¸ï¼æ¯ä¸ªåæ¥åçç»å°¾åæ¥äºï¼synchronize withï¼ä¸ä¸ªåæ¥åçå¼å§ãå åµäºå ¶ä»åæ¥åç忥忲¡æç¹æ®è¯ä¹ã
忥å䏿¯äºå¡ï¼ä¸åäºåé¢çåååï¼ï¼å¹¶å¯ä»¥è°ç¨äºå¡ä¸å®å ¨ç彿°ã
#include <iostream> #include <vector> #include <thread> int f() { static int i = 0; synchronized { // å¼å§åæ¥å std::cout << i << " -> "; ++i; // æ¯æ¬¡è°ç¨ f() é½è·å¾ i çå¯ä¸å¼ std::cout << i << '\n'; return i; // ç»æåæ¥å } } int main() { std::vector<std::thread> v(10); for(auto& t: v) t = std::thread([]{ for(int n = 0; n < 10; ++n) f(); }); for(auto& t: v) t.join(); }
è¾åºï¼
0 -> 1 1 -> 2 2 -> 3 ... 99 -> 100
以任使¹å¼ï¼æµè¾¾ç»å°¾ï¼æ§è¡ gotoãbreakãcontinue æ returnï¼ææåºå¼å¸¸ï¼ç¦»å¼åæ¥åé½ä¼éåºè¯¥åï¼èè¥æéåºç忝å¤å±åï¼åè¿å¨åä¸å ¨åºä¸åæ¥äºä¸ä¸ªåæ¥åãè¥ä½¿ç¨ std::longjmp éåºåæ¥ååè¡ä¸ºæªå®ä¹ã
ä¸å è®¸ç¨ goto æ switch è¿å ¥åæ¥åã
å°½ç®¡åæ¥åå¦åå¨ä¸ä¸ªå
¨å±é䏿§è¡ï¼æä»¬ä»ç¶æå¾
åå®ç°æ£éªæ¯ä¸ªåå
ç代ç ï¼å¹¶ä¸ºäºå¡å®å
¨ä»£ç 使ç¨ä¹è§å¹¶åï¼å¨å¯ç¨æ¶ä»¥ç¡¬ä»¶äºå¡æ§å
å为åç¾ï¼ï¼ä¸ºéäºå¡å®å
¨ä»£ç ä½¿ç¨æå°éå®ãå½åæ¥åè°ç¨éå
è彿°æ¶ï¼é¤éè¯¥å½æ°å£°æä¸º transaction_safeï¼è§ä¸æï¼æä½¿ç¨ [[optimize_for_synchronized]] 屿§ï¼è§ä¸æï¼ï¼å¦åç¼è¯å¨å¯è½å¿
é¡»æ¾å¼æ¨æµæ§è¡ï¼spculative executionï¼ï¼å¹¶å¨æ´ä¸ªè°ç¨å¨å´ææä¸ä¸ªéã
ååå
| æ¬èæªå®æ |
atomic_noexcept å¤åè¯å¥
atomic_cancel å¤åè¯å¥
atomic_commit å¤åè¯å¥
ç¨äº atomic_cancel åä¸çäºå¡åæ¶çå¼å¸¸æ std::bad_allocãstd::bad_array_new_lengthãstd::bad_castãstd::bad_typeidãstd::bad_exceptionãstd::exception åææä»å®æ´¾ççæ ååºå¼å¸¸ï¼ä»¥åç¹æ®å¼å¸¸ç±»å std::tx_exception<T>ã
ä¸å
许åååä¸ç å¤åè¯å¥ æ§è¡ä»»ä½é transaction_safe çè¡¨è¾¾å¼æè¯å¥ï¼æè°ç¨é transaction_safe ç彿°ï¼è¿æ¯ç¼è¯æ¶é误ï¼ã
// æ¯æ¬¡è°ç¨ f() é½åå¾ i çå¯ä¸å¼ï¼å³ä½¿ä»¥å¹¶è¡è¿è¡ int f() { static int i = 0; atomic_noexcept { // å¼å§äºå¡ // printf("before %d\n", i); // é误ï¼ä¸è½è°ç¨éäºå¡å®å ¨ç彿° ++i; return i; // æäº¤äºå¡ } }
以é¤å¼å¸¸ä¹å¤ç任使¹å¼ï¼æµè¾¾ç»å°¾ãgotoãbreakãcontinueãreturnï¼ç¦»å¼åååæ¶ï¼å°æäº¤äºå¡ãè¥ç¨ std::longjmp éåºååååè¡ä¸ºæªå®ä¹ã
äºå¡å®å ¨ç彿°
| æ¬èæªå®æ |
å¯å¨å½æ°å£°æä¸ç¨å ³é®è¯ transaction_safe å°å ¶æ¾å¼å£°æä¸ºäºå¡å®å ¨ã
| æ¬èæªå®æ |
mutable ä¹åï¼è¥ä½¿ç¨å®ï¼åºç°ã
| æ¬èæªå®æ |
extern volatile int * p = 0; struct S { virtual ~S(); }; int f() transaction_safe { int x = 0; // OKï¼é volatile p = &x; // OKï¼æéé volatile int i = *p; // é误ï¼éè¿ volatile æ³å·¦å¼è¯»å S s; // é误ï¼è°ç¨ä¸å®å ¨çææå½æ° }
int f(int x) { // éå¼äºå¡å®å ¨ if (x <= 0) return 0; return x + f(x-1); }
è¥éè¿æåäºå¡å®å ¨å½æ°çå¼ç¨ææéè°ç¨éäºå¡å®å ¨ç彿°ï¼åè¡ä¸ºæªå®ä¹ã
æåäºå¡å®å
¨å½æ°çæéåæåäºå¡å®å
¨æå彿°çæéåå«å¯éå¼è½¬æ¢ä¸ºå½æ°æéåæå彿°æéãç»ææéååæéæ¯å¦æ¯è¾ç¸çæ¯æªææçã
äºå¡å®å ¨çè彿°
| æ¬èæªå®æ |
è¥ transaction_safe_dynamic 彿°çæç»è¦ç彿°æªè¢«å£°æä¸º transaction_safeï¼åå¨åååä¸è°ç¨å®æ¯æªå®ä¹è¡ä¸ºã
æ ååº
é¤äºå¼å ¥æ°çå¼å¸¸æ¨¡æ¿ std::tx_exception ä¹å¤ï¼äºå¡æ§å åææ¯è§èè¿å¯¹æ ååºååºä¸åæ´æ¹ï¼
- 令ä¸å彿°ä¸ºæ¾å¼
transaction_safeï¼
- std::forwardãstd::moveãstd::move_if_noexceptãstd::alignãstd::abortãå
¨å±é»è®¤ operator newãå
¨å±é»è®¤ operator deleteãstd::allocator::constructï¼è¥è°ç¨çæé 彿°ä¸ºäºå¡å®å
¨ï¼ãstd::allocator::destroyï¼è¥è°ç¨çææå½æ°ä¸ºäºå¡å®å
¨ï¼ãstd::get_temporary_bufferãstd::return_temporary_bufferãstd::addressofãstd::pointer_traits::pointer_toãæææ¯æäºå¡åæ¶çå¼å¸¸ç±»åï¼è§ä¸æç
atomic_cancelï¼çæ¯ä¸ªéèæå彿°æ¬èæªå®æ
åå ï¼è¿ææ´å¤
- std::forwardãstd::moveãstd::move_if_noexceptãstd::alignãstd::abortãå
¨å±é»è®¤ operator newãå
¨å±é»è®¤ operator deleteãstd::allocator::constructï¼è¥è°ç¨çæé 彿°ä¸ºäºå¡å®å
¨ï¼ãstd::allocator::destroyï¼è¥è°ç¨çææå½æ°ä¸ºäºå¡å®å
¨ï¼ãstd::get_temporary_bufferãstd::return_temporary_bufferãstd::addressofãstd::pointer_traits::pointer_toãæææ¯æäºå¡åæ¶çå¼å¸¸ç±»åï¼è§ä¸æç
- 令ä¸å彿°ä¸ºæ¾å¼
transaction_safe_dynamic
- æææ¯æäºå¡åæ¶çå¼å¸¸ç±»åï¼è§ä¸æç
atomic_cancelï¼çæ¯ä¸ªè彿°
- æææ¯æäºå¡åæ¶çå¼å¸¸ç±»åï¼è§ä¸æç
- è¦æ±åé
å¨ (Allocator) X 䏿æäºå¡å®å
¨çæä½å¨
X::rebind<>::otherä¸äº¦ä¸ºäºå¡å®å ¨
屿§
[[optimize_for_synchronized]] 屿§å¯åºç¨å°å½æ°å£°æä¸ç声æç¬¦ä¸ï¼èä¸å¿
é¡»å¨å½æ°çé¦ä¸ªå£°æä¸åºç°ã
è¥æå½æ°å¨ä¸ä¸ªç¿»è¯åå
ä¸è¢«å£°æä¸º [[optimize_for_synchronized]]ï¼èåä¸ä¸ªå½æ°å¨å¦ä¸ç¿»è¯åå
ä¸ç声æä¸å¸¦ [[optimize_for_synchronized]]ï¼åç¨åºéè¯æï¼ä¸è¦æ±è¯æã
å®æç¤ºå½æ°å®ä¹åºè¯¥éå¯¹ä» synchronized è¯å¥ä¸è°ç¨èä¼åãå ·ä½èè¨ï¼è¥å½æ°å¯¹å¤§å¤æ°çè°ç¨ä½éå ¨é¨è°ç¨ä¸ºäºå¡å®å ¨ï¼ä¾å¦å¯è½å¿ é¡»éç®æ£åçåå¸è¡¨æå ¥ãå¯è½å¿ é¡»è¯·æ±æ°å ååçåé å¨ãå¯è½ç½æè®°å½æ¥å¿çç®å彿°ï¼ï¼åè¯¥å±æ§é¿å 对è°ç¨è¯¥å½æ°ç忥åè¿è¡ä¸²è¡åã
std::atomic<bool> rehash{false};  // ç»´æ¤çº¿ç¨è¿è¡æ¤å¾ªç¯ void maintenance_thread(void*) { while (!shutdown) { synchronized { if (rehash) { hash.rehash(); rehash = false; } } } }  // å·¥ä½çº¿ç¨æ¯ç§æ§è¡æ°å䏿¬¡å¯¹æ¤å½æ°çè°ç¨ã // ä»å¦ä¸ç¿»è¯åå ä¸ç忥åè°ç¨ insert_key() å°å¯¼è´è¿äºå被串è¡åï¼ // é¤éæ è®° insert_key() 为 [[optimize_for_synchronized]] [[optimize_for_synchronized]] void insert_key(char* key, char* value) { bool concern = hash.insert(key, value); if (concern) rehash = true; }
æ è¯¥å±æ§ç GCC æ±ç¼ï¼ä¸²è¡åæ´ä¸ªå½æ°
insert_key(char*, char*): subq $8, %rsp movq %rsi, %rdx movq %rdi, %rsi movl $hash, %edi call Hash::insert(char*, char*) testb %al, %al je .L20 movb $1, rehash(%rip) mfence .L20: addq $8, %rsp ret
æè¯¥å±æ§ç GCC æ±ç¼ï¼
transaction clone for insert_key(char*, char*): subq $8, %rsp movq %rsi, %rdx movq %rdi, %rsi movl $hash, %edi call transaction clone for Hash::insert(char*, char*) testb %al, %al je .L27 xorl %edi, %edi call _ITM_changeTransactionMode # 注æï¼è¿æ¯ä¸²è¡åç¹ movb $1, rehash(%rip) mfence .L27: addq $8, %rsp ret
| æ¬èæªå®æ åå ï¼æ£æ¥å¸¦ä¸»å¹²çæ±ç¼ï¼äº¦æ¾ç¤ºè°ç¨æ¹çæ´æ¹ |
注解
| æ¬èæªå®æ åå ï¼æ¥èª Wyatt ææ¡/讨论çä½éªç¬è®° |
ç¼è¯å¨æ¯æ
GCC ä»çæ¬ 6.1 èµ·æ¯ææ¤ææ¯è§èï¼è¦æ±å¯ç¨ -fgnu-tmï¼ãGCC 4.7 æ¶æ¾æ¯ææ¤è§èçä¸ä¸ªæ§çåä½ã