c++ primer 第五版學(xué)習(xí)筆記

第二章

  • 函數(shù)體外定義的內(nèi)置類型變量會初始化為0,函數(shù)體外的是未初始化的
  • constexpr聲明變量表示它是一個常量表達(dá)式(編譯器可以確定的值),且只能應(yīng)用于字面值
  • c++11中可以用using SI = Sales_item 來定義一個類型別名
  • top-level是指復(fù)合類型(指針、引用)對象本身是const,low-level是指復(fù)合類型所指的對象是const
  • delctype里對變量使用(),返回的推導(dǎo)類型永遠(yuǎn)是引用。比如
    int i;
    delctype((i)) d; // d是&int
    delctype(i) e; // e是int
    
  • c++11中,可以為在類的聲明中直接給類的成員賦予默認(rèn)值:
    class item
    {
    private:
        int data = 1;
    };
    

第三章

  • vector可以用初始化列表來進(jìn)行初始化vector<int> ivec{1,2,3};

    如果編譯器無法用{}內(nèi)的內(nèi)容進(jìn)行列表初始化,將會嘗試用()的方式來進(jìn)行初始化。
    
    ```vector<string> v8{10, "hi"}; // v8 has ten elements with value "hi"```
    
  • range for瀏覽容器時,不能改變?nèi)萜鞯拇笮?/strong>

  • c++11中加入了cbegincend來返回對應(yīng)beginendconst迭代器

  • 列表初始化數(shù)組時,如果數(shù)組的維度大于列表中的元素個數(shù),剩余元素會用默認(rèn)值初始化

  • delctype作用于數(shù)組時,返回的也是數(shù)組本身的類型:

    decltype(ia) ia3 = {0,1,2,3,4,5,6,7,8,9};
    
  • c++11中可以用包含在頭文件iterator中的beginend函數(shù),對數(shù)組求得對應(yīng)位置的指針:

    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
    
  • 數(shù)組下標(biāo)小于0時,按照0取值

第四章

  • c++11中, m/n + m%n一定等于m,所以m%n的結(jié)果也可能是負(fù)的

第五章

  • 新標(biāo)準(zhǔn)中,只要一個類型可以被move,就可以創(chuàng)建這種類型的容器
  • 不能把一個右值引用綁定到一個變量上,即使這個變量本身就是個右值引用
  • 不拋出異常的move的構(gòu)造函數(shù)和賦值操作符應(yīng)該標(biāo)注為noexcept。
  • 只有當(dāng)一個類沒有定義任何復(fù)制控制成員并且僅當(dāng)所有成員數(shù)據(jù)都可以分別被move構(gòu)造和賦值時,編譯器才會為其合成move構(gòu)造函數(shù)和賦值操作符
  • 定義了move構(gòu)造函數(shù)和賦值操作符的類必須定義自己的復(fù)制操作,否則復(fù)制成員默認(rèn)情況下會被刪掉
  • 可以在成員函數(shù)的形參列表后加&&& 符號來限定this是左值引用或右值引用,如果同時是const的,則const在前
  • 如果一個成員函數(shù)有引用限定符,所有一樣形參的重載版本成員函數(shù)必須都具有引用限定符。

第六章

  • Initializer_list是一個標(biāo)準(zhǔn)庫類型,可以用來在函數(shù)中傳遞可變個數(shù)的參數(shù)列表,也可以作為參數(shù)的返回值
  • 可以用下面的形式來定義函數(shù)返回類型:
    auto func(int i) -> int(*)[10];
    decltype(odd) *arrPtr(int i)
    
  • c++11里增加了constexpr函數(shù),在聲明前加constexpr即可。

第七章

  • c++11中在成員函數(shù)聲明或定義后加= default 代表讓編譯器生成其默認(rèn)版本。
  • c++11中可以使用初始化列表給類成員添加默認(rèn)值
    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里可以使用代理構(gòu)造函數(shù)
    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構(gòu)造函數(shù)

第九章

  • emplace_front, emplace, emplace_back分別對應(yīng)push_front, insert,push_back,用來在往容器中插入一個新構(gòu)造的元素
  • 為了使操作在常量時間完成,forward_list沒有insert,emplace,erase,而是定義了insert_after, emplace_after,erase_after以及before_begin.
  • 可以使用shrink_to_fit來嘗試回收容器沒有使用的內(nèi)存,但并不保證
  • c++11增加了字符串和內(nèi)置類型互轉(zhuǎn)的接口:
    • to_string(val) 轉(zhuǎn)為string
    • sto(s) 轉(zhuǎn)為所代表的類型簡寫,比如i l ul ull f d ld等

第十章

  • 標(biāo)準(zhǔn)庫算法從不改變?nèi)萜鞯拇笮?/strong>
  • lambda表達(dá)式的基本形式:[capture list] (parameter list) -> return type { **function** body }
    • [&] [=] 分別表示lambda里使用的局部變量隱式地以引用和傳值的形式被捕獲。
    • [&, identifier_list] 表示identifier_list里的變量是傳值捕獲,其他變量以引用形式捕獲
    • [=, reference_list] 表示reference_list里的變量是引用捕獲,其他變量以傳值形式捕獲
    • 如果需要在lambda里修改以傳值捕獲的變量,需要在函數(shù)體前加上mutable
    • bind中使用ref(objcref(obj)可以將obj以普通引用或const引用的方式傳遞

第十二章

  • 絕對不要用shared_ptrget接口來初始化或賦值給其他shared_ptr
  • 盡量不要把shared_ptr管理的資源以內(nèi)置指針的形式暴露出來
  • 如果使用智能指針管理的資源不是new出來的,記得要給它傳遞一個deleter

第十三章

  • 如果一個類需要析構(gòu)函數(shù),那么也幾乎也確定需要復(fù)制構(gòu)造函數(shù)和賦值操作符
  • c++11中可以在給類的復(fù)制控制函數(shù)和構(gòu)造函數(shù)后加= default來讓編譯器生成一個默認(rèn)版本。該函數(shù)是否內(nèi)聯(lián)取決于= default加在聲明后還是定義后。
  • c++11中可以在類的復(fù)制控制函數(shù)和構(gòu)造函數(shù)后加= delete來讓編譯器禁止生成其對應(yīng)版本,它必須出現(xiàn)在聲明之后。
  • 變量表達(dá)式本身是一個左值,例如:
    int &&rr1 = 42; // ok: literals are rvalues
    int &&rr2 = rr1; // error: the expression rr1 is an lvalue!
    
  • 被使用move的對象可以被賦值或析構(gòu),但不能繼續(xù)使用它的值
  • 編譯器僅為那些沒有定義任何復(fù)制控制成員并且所有成員都可以被move的類合成move構(gòu)造函數(shù)或move賦值操作符。
  • 當(dāng)一個類滿足一下幾個條件之一時,編譯器無法合成move控制函數(shù):
    • 成員擁有復(fù)制控制函數(shù)卻沒有move控制函數(shù)
    • 有成員的move控制函數(shù)或是deleted或者不可訪問的
    • 析構(gòu)函數(shù)是deleted或者不可訪問的
    • 有成員是const或者引用時
  • 一個類定義了move控制函數(shù)(構(gòu)造、賦值)時,也應(yīng)該定義復(fù)制控制函數(shù),否則他們默認(rèn)情況下會被定義為deleted
  • make_move_iterator函數(shù)接受一個普通迭代器,返回一個move迭代器。對move迭代器解引用產(chǎn)生一個右值引用。
  • 定義const成員函數(shù)時可以在參數(shù)列表后面加上&&&表明返回值的左/右值屬性,需要跟在const之后(如果有的話)
  • 為一個類定義超過一個同名同參的成員函數(shù)時,要么全部提供引用限定符,要么一個也不提供。

第十四章

  • c++11中的function類可以用來定義一個函數(shù)對象,例如:

    function<int(int, int)> f1 = add;

    定義了名為f1,返回類型為int,2個int參數(shù)的函數(shù)對象,調(diào)用函數(shù)為add函數(shù)。

  • c++11中類型轉(zhuǎn)換函數(shù)可以加explicit關(guān)鍵字來表明只有顯示調(diào)用,才可以產(chǎn)生類型轉(zhuǎn)換。

第十五章

  • c++11增加了override用來顯式地表明此函數(shù)覆蓋基類的虛函數(shù)
  • c++11中類名后加上final表明這個類不能作為基類,在成員函數(shù)后加表明子類不能覆蓋它

第十六章

  • c++11中可以用template<typename T> using twin = pair<T, T>;來為模版類型定義別名。

  • c++11中,模版類和模板函數(shù)都可以使用默認(rèn)值

  • c++11中,可以使用extern template declaration; 來聲明一個模版實例,這樣可以減少編譯器生成的模板實例。

  • 對于模板類型參數(shù),只有const轉(zhuǎn)換和數(shù)組或函數(shù)指針的轉(zhuǎn)換會自動進(jìn)行。對于非模板類型的參數(shù),則進(jìn)行正常的自動轉(zhuǎn)換。

  • 類型別名或模版參數(shù)會導(dǎo)致引用的引用產(chǎn)生引用折疊,規(guī)則如下:

    X& &, X& &&, and X&& & all collapse to type X&

    X&& && collapses to X&&

  • c++11中,std::forward可以實現(xiàn)完美轉(zhuǎn)發(fā),即保留參數(shù)的左右值屬性及const屬性。

  • 假設(shè)有2個模板函數(shù)

    1. template<typename T> func(**const**  T &t) 
    2. template<typename T> func(**const** T* t)
    

則對于一個const T* p的參數(shù),編譯器會優(yōu)先調(diào)用第二個。因為第二個更specialized,第一個更general。

  • c++11中,結(jié)合可變參數(shù)模板加右值引用和std::forward實現(xiàn)可變參數(shù)完美轉(zhuǎn)發(fā):
    // 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)...);
    }
    
  • 對于可變參數(shù)列表,可以使用某些格式使其展開。但展開后以逗號分隔的參數(shù)只能作為函數(shù)調(diào)用實參而不是逗號運算符。例子:
    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,顯示整數(shù)時會加上0或0x
  • c++11里可以使用scientific操作符,使得實數(shù)以科學(xué)計數(shù)法表示
  • noskipws操作符可以使輸入流讀取空格

第十八章

  • c++11中,可以于函數(shù)的聲明或定義時最后跟上noexcept表明此函數(shù)不會拋出異常。也可以帶一個bool參數(shù),為false時表示可以拋出異常。
  • noexcept也可以作為一個表達(dá)式,返回作為參數(shù)傳入的函數(shù)調(diào)用鏈中是否會拋出異常。
  • c++11中,可以在namespace聲明前加inline,這樣該namespace中的符號可以直接使用而不需要加上****namespace::
  • 可以用 namespace primer = cplusplus_primer; 這樣的方式給命名空間cplusplus_primer起別名

第十九章

  • c++11中增加了scoped枚舉類型,在enum后面加structclass表明。例如:

    enum class peppers {red, yellow, green};

    這個枚舉在使用時需要加上域名peppers,并且它不能自動轉(zhuǎn)換為int類型。

  • c++11中,可以在枚舉名后指定需要的表示類型:

    enum intValues : unsigned long long 
    {
        longTyp = 4294967295UL,
        long_longTyp = 18446744073709551615ULL
    }
    
  • c++11中加入了枚舉類型的前向聲明,需要指定名字和實現(xiàn)類型。

  • c++11中加入了mem_fn,與function用法類似,可以用來綁定類的成員函數(shù)。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容