第二章
- 函數體外定義的內置類型變量會初始化為0,函數體外的是未初始化的
- 用constexpr聲明變量表示它是一個常量表達式(編譯器可以確定的值),且只能應用于字面值
-
c++11中可以用
using SI = Sales_item
來定義一個類型別名 - top-level是指復合類型(指針、引用)對象本身是const,low-level是指復合類型所指的對象是const
- 在delctype里對變量使用(),返回的推導類型永遠是引用。比如
int i; delctype((i)) d; // d是&int delctype(i) e; // e是int
-
c++11中,可以為在類的聲明中直接給類的成員賦予默認值:
class item { private: int data = 1; };
第三章
-
vector可以用初始化列表來進行初始化
vector<int> ivec{1,2,3};
如果編譯器無法用{}內的內容進行列表初始化,將會嘗試用()的方式來進行初始化。 ```vector<string> v8{10, "hi"}; // v8 has ten elements with value "hi"```
range for瀏覽容器時,不能改變容器的大小
c++11中加入了cbegin和cend來返回對應begin和end的const迭代器
列表初始化數組時,如果數組的維度大于列表中的元素個數,剩余元素會用默認值初始化
-
delctype作用于數組時,返回的也是數組本身的類型:
decltype(ia) ia3 = {0,1,2,3,4,5,6,7,8,9};
-
c++11中可以用包含在頭文件iterator中的begin和end函數,對數組求得對應位置的指針:
int ia[] = {0,1,2,3,4,5,6,7,8,9}; // ia is an array of ten ints int *beg = begin(ia); // pointer to the first element in ia int *last = end(ia); // pointer one past the last element in ia
數組下標小于0時,按照0取值
第四章
-
c++11中,
m/n + m%n
一定等于m,所以m%n
的結果也可能是負的
第五章
- 新標準中,只要一個類型可以被move,就可以創建這種類型的容器
- 不能把一個右值引用綁定到一個變量上,即使這個變量本身就是個右值引用
- 不拋出異常的move的構造函數和賦值操作符應該標注為noexcept。
- 只有當一個類沒有定義任何復制控制成員并且僅當所有成員數據都可以分別被move構造和賦值時,編譯器才會為其合成move構造函數和賦值操作符
- 定義了move構造函數和賦值操作符的類必須定義自己的復制操作,否則復制成員默認情況下會被刪掉
- 可以在成員函數的形參列表后加
&
或&&
符號來限定this是左值引用或右值引用,如果同時是const的,則const在前 - 如果一個成員函數有引用限定符,所有一樣形參的重載版本成員函數必須都具有引用限定符。
第六章
- Initializer_list是一個標準庫類型,可以用來在函數中傳遞可變個數的參數列表,也可以作為參數的返回值
- 可以用下面的形式來定義函數返回類型:
auto func(int i) -> int(*)[10]; decltype(odd) *arrPtr(int i)
- c++11里增加了constexpr函數,在聲明前加constexpr即可。
第七章
-
c++11中在成員函數聲明或定義后加
= default
代表讓編譯器生成其默認版本。 -
c++11中可以使用初始化列表給類成員添加默認值
class Window_mgr { private: // Screens this Window_mgr is tracking // by default, a Window_mgr has one standard sized blank Screen std::vector<Screen> screens{Screen(24, 80, ' ') }; };
-
c++11里可以使用代理構造函數
class Sales_data { public: // nondelegating constructor initializes members from corresponding arguments Sales_data(std::string s, unsigned cnt, double price): bookNo(s), units_sold(cnt), revenue(cnt*price) {} // remaining constructors all delegate to another constructor Sales_data(): Sales_data("", 0, 0) {} Sales_data(std::string s): Sales_data(s, 0,0) {} Sales_data(std::istream &is): Sales_data() { read(is, *this); } // other members as before };
? c++11中有constexpr類,其必須提供至少一個constexpr構造函數
第九章
- emplace_front, emplace, emplace_back分別對應push_front, insert,push_back,用來在往容器中插入一個新構造的元素
- 為了使操作在常量時間完成,forward_list沒有insert,emplace,erase,而是定義了insert_after, emplace_after,erase_after以及before_begin.
- 可以使用shrink_to_fit來嘗試回收容器沒有使用的內存,但并不保證
-
c++11增加了字符串和內置類型互轉的接口:
- to_string(val) 轉為string
- sto(s) 轉為所代表的類型簡寫,比如i l ul ull f d ld等
第十章
- 標準庫算法從不改變容器的大小
-
lambda表達式的基本形式:
[capture list] (parameter list) -> return type { **function** body }
- [&] [=] 分別表示lambda里使用的局部變量隱式地以引用和傳值的形式被捕獲。
- [&, identifier_list] 表示identifier_list里的變量是傳值捕獲,其他變量以引用形式捕獲
- [=, reference_list] 表示reference_list里的變量是引用捕獲,其他變量以傳值形式捕獲
- 如果需要在lambda里修改以傳值捕獲的變量,需要在函數體前加上mutable
- 在bind中使用ref(obj或cref(obj)可以將obj以普通引用或const引用的方式傳遞
第十二章
- 絕對不要用shared_ptr的get接口來初始化或賦值給其他shared_ptr
- 盡量不要把shared_ptr管理的資源以內置指針的形式暴露出來
- 如果使用智能指針管理的資源不是new出來的,記得要給它傳遞一個deleter
第十三章
- 如果一個類需要析構函數,那么也幾乎也確定需要復制構造函數和賦值操作符
-
c++11中可以在給類的復制控制函數和構造函數后加
= default
來讓編譯器生成一個默認版本。該函數是否內聯取決于= default
加在聲明后還是定義后。 -
c++11中可以在類的復制控制函數和構造函數后加
= delete
來讓編譯器禁止生成其對應版本,它必須出現在聲明之后。 - 變量表達式本身是一個左值,例如:
int &&rr1 = 42; // ok: literals are rvalues int &&rr2 = rr1; // error: the expression rr1 is an lvalue!
- 被使用move的對象可以被賦值或析構,但不能繼續使用它的值
- 編譯器僅為那些沒有定義任何復制控制成員并且所有成員都可以被move的類合成move構造函數或move賦值操作符。
- 當一個類滿足一下幾個條件之一時,編譯器無法合成move控制函數:
- 成員擁有復制控制函數卻沒有move控制函數
- 有成員的move控制函數或是deleted或者不可訪問的
- 析構函數是deleted或者不可訪問的
- 有成員是const或者引用時
- 一個類定義了move控制函數(構造、賦值)時,也應該定義復制控制函數,否則他們默認情況下會被定義為deleted
- make_move_iterator函數接受一個普通迭代器,返回一個move迭代器。對move迭代器解引用產生一個右值引用。
- 定義const成員函數時可以在參數列表后面加上
&
或&&
表明返回值的左/右值屬性,需要跟在const之后(如果有的話) - 為一個類定義超過一個同名同參的成員函數時,要么全部提供引用限定符,要么一個也不提供。
第十四章
-
c++11中的function類可以用來定義一個函數對象,例如:
function<int(int, int)> f1 = add;
定義了名為f1,返回類型為int,2個int參數的函數對象,調用函數為add函數。
c++11中類型轉換函數可以加explicit關鍵字來表明只有顯示調用,才可以產生類型轉換。
第十五章
- c++11增加了override用來顯式地表明此函數覆蓋基類的虛函數
- c++11中類名后加上final表明這個類不能作為基類,在成員函數后加表明子類不能覆蓋它
第十六章
c++11中可以用
template<typename T> using twin = pair<T, T>;
來為模版類型定義別名。c++11中,模版類和模板函數都可以使用默認值
c++11中,可以使用
extern template declaration;
來聲明一個模版實例,這樣可以減少編譯器生成的模板實例。對于模板類型參數,只有const轉換和數組或函數指針的轉換會自動進行。對于非模板類型的參數,則進行正常的自動轉換。
-
類型別名或模版參數會導致引用的引用產生引用折疊,規則如下:
X& &, X& &&, and X&& & all collapse to type X&
X&& && collapses to X&&
c++11中,std::forward可以實現完美轉發,即保留參數的左右值屬性及const屬性。
-
假設有2個模板函數
1. template<typename T> func(**const** T &t) 2. template<typename T> func(**const** T* t)
則對于一個const T* p的參數,編譯器會優先調用第二個。因為第二個更specialized,第一個更general。
-
c++11中,結合可變參數模板加右值引用和std::forward實現可變參數完美轉發:
// an rvalue reference to a template parameter type template<typename... Args> void fun(Args&&... args) // expands Args as a list of rvalue references { // the argument to work expands both Args and args work(std::forward<Args>(args)...); }
- 對于可變參數列表,可以使用某些格式使其展開。但展開后以逗號分隔的參數只能作為函數調用實參而不是逗號運算符。例子:
template <typename T> void print(**const** T& t) { cout << t << " "; } template <typename ...Args> void varadic_print(Args&&... args) { int arr[]{(print(args), 0)...}; cout << endl; } varadic_print(1, "abc", 2.0); //輸出 1 abc 2
第十七章
- 輸出流可以使用操作符showbase,顯示整數時會加上0或0x
- c++11里可以使用scientific操作符,使得實數以科學計數法表示
- noskipws操作符可以使輸入流讀取空格
第十八章
- c++11中,可以于函數的聲明或定義時最后跟上noexcept表明此函數不會拋出異常。也可以帶一個bool參數,為false時表示可以拋出異常。
- noexcept也可以作為一個表達式,返回作為參數傳入的函數調用鏈中是否會拋出異常。
- c++11中,可以在namespace聲明前加inline,這樣該namespace中的符號可以直接使用而不需要加上****namespace::
- 可以用
namespace primer = cplusplus_primer;
這樣的方式給命名空間cplusplus_primer起別名
第十九章
-
c++11中增加了scoped枚舉類型,在enum后面加struct或class表明。例如:
enum class peppers {red, yellow, green};
這個枚舉在使用時需要加上域名peppers,并且它不能自動轉換為int類型。
-
在c++11中,可以在枚舉名后指定需要的表示類型:
enum intValues : unsigned long long { longTyp = 4294967295UL, long_longTyp = 18446744073709551615ULL }
c++11中加入了枚舉類型的前向聲明,需要指定名字和實現類型。
c++11中加入了mem_fn,與function用法類似,可以用來綁定類的成員函數。