• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            posts - 13, comments - 4, trackbacks - 0, articles - 0
              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

            Exceptional C++ Style 讀書筆記(一)

            Posted on 2008-11-03 20:36 Batiliu 閱讀(481) 評論(0)  編輯 收藏 引用 所屬分類: 讀書筆記

            第一條 vector的使用

            設計vector是用來代替內置數組的,因此其效率應該與內置數組一樣,內置數組在下標索引(operator[])時是不進行越界檢查的。如果你需要下標越界檢查,可以使用at方法。

             

            準則記住size/resize以及capacity/reserve之間的區別。
            • size告訴你容器中目前實際有多少個元素,而對應地,resize則會在容器的尾部添加或刪除一些元素。這兩個函數對list、vector、deque都適用,但對其他容器并不適用。
            • capacity則告訴你最少添加多少個元素才會導致容器重分配內存,而reserve在必要的時候總是會使容器的內部緩沖區擴充至一個更大的容量,以確保至少能滿足你所指出的空間大小。這兩個函數僅對vector適用。

             

             

            準則

            • 記得使用const_iterator。
            • 盡量使用!=而不是<來比較兩個iterator。
            • 養成默認情況下使用前綴形式的--和++的習慣,除非你的確需要用到原來的值。
            • 盡量服用標準庫中已有的算法,例如for_each()。

             

             

            第二條 第三條 字符串格式化

                   sprintf     snprintf     stringstream     strstream     boost::lexical_cast    
            易用嗎,代碼清晰明確嗎
            高效嗎,無額外內存分配嗎
            長度安全嗎
            類型安全嗎
            是否可以用于模板之中
                    是            是               否                    否                      是
                    是            是               否                    是                      否
                    否            是               是                    是                      是
                    否            否               是                    是                      是
                    否            否               是                    是                      是

             

            準則

            • 如果你所要做的只是將一個值轉換為一個字符串的話,盡量默認使用boost::lexical_cast。
            • 如果想進行簡單的格式化,或者需要支持寬字符串,或者在模板當中進行格式化,盡量使用stringstream或strstream。
            • 如果想進行一些復雜的格式化,停駛并不需要支持寬字符串,也不想讓代碼用在模板當中的話,盡量使用snprintf。
            • 永遠不要使用sprintf。

             

            第四條 標準庫成員函數

            使用標準庫里面的mem_fun,將成員函數適配為仿函數(functor),從而可被標準庫算法以及其他正常情況下只適用自由函數的代碼所使用。但不要將它用在標準庫自己身上。

             

            第五條 第六條 泛型性的風味

             

            準則如果你的類型有一個更好的途徑可以用于交換其對象的值的話,請考慮為它特化std::swap()。

             

            // 示例:重載swap()。
             
            class X {
            public:
                void swap(X&);
            };
             
            // 注意:這個重載并沒有放在std名字空間當中。請參考名字空間Koenig查找。
            swap(X&a, X&b) {
                a.swap(b);
            }

             

            第七條 泛型性的風味

             

            準則記住,函數模板不能偏特化,只能重載。寫一個看似函數模板偏特化的函數模板實際上是在寫一個單獨的主函數模板。

             

             

            在如下代碼中,最后一行代碼調用的是f()的那個版本?

            // 示例:函數特化。
             
            template <typename T>    // (a):一個主模板
            void f(T);
             
            template <typename T>    // (b):一個主模板,重載了(a)
            void f(T*);
             
            template <>              // (c):(a)的一個顯示特化(全特化)
            void f<int*>(int*);
             
            template <>              // (d):(b)的一個顯示特化(全特化)
            void f<int>(int*);
             
            // ....
             
            int * p;
            f(p);                    // 調用的哪個版本?

            答案是……調用(d),想必結果是你意料之中。如果我們注釋掉(d),f()調用的又是哪個版本呢?答案是……調用(b),是不是讓你大吃一驚?如果這令你感到驚訝的話,你并不是唯一一個,當時它曾讓許多專家大吃一驚。理解這個例子的關鍵其實很簡單:模板特化并不參加重載。

             

             

            準則記住,函數模板特化并不參與重載決議。只有在某個主模板被重載決議選中的前提下,其特化版本才有可能被使用。而且,編譯器在選擇主模板的時候并不關心它是否有某個特化版本。

             

            讓我們來看看函數模板的重載規則。

            • 非模板函數是C++中的一等公民。如果一個普通的非模板函數跟一個函數模板在重載解析的參數匹配中表現一樣好的話,編譯器會選擇普通函數。
            • 如果編譯器沒有發現合適的一等公民,那么主函數模板作為C++中的二等公民就會被納入考慮。具體選擇哪個主函數模板則取決于哪個的參數類型匹配的最好。
            • 如果這個被選中的主函數模板碰巧有針對所使用的模板實參做了特化的話,該特化版本才會被編譯器選中。否則編譯器將使用以正確類型實例化的主模板。
            • 等等

             

             

            準則如果你想要將一個主模板特化,同時又希望該特化版本能夠參與重載決議(或希望確保當它完全匹配用戶調用時能被編譯器選中)的話,只需將其寫成普通函數即可。

            推論:如果你確實提供了某個函數模板的重載,那么你應當避免為它提供特化。

             

             

            準則如果你正在寫一個可能需要被特化的主函數模板的話,請盡量將它寫成一個孤立的、永遠不該被特化或重載的函數模板,并將其具體實現全部放入一個包含了一個靜態函數的類模板當中。這么一來任何人都可以對后者進行特化(全特化或偏特化),而同時又不會影響到主函數模板的重載決議。

             

            // 示例:函數特化。
             
            template <typename T>
            struct X;
             
            template <typename T>            // 請不要動這個函數
            void f(T t) { X<T>::far(t); }
             
            template <typename T>            // 可以對這個函數進行特化
            struct X {
                static void f(T t);
            };

             

            第八條 友元模板

            準則明確地表達出你的意圖。如果你所指的確實是一個模板,可以在模板名字后面加上模板實參列表(可以為空,即<>)。

             

            // 示例:友元模板。
             
            namespace boost {
                template <typename T> void checked_delete(T* x);
            }
             
            class Test {
                friend void boost::checked_delete<Test>(Test* x);    // 或"checked_delete<>(Test* x)"
            }

             

            第九條 第十條 導出限制

            準則記住,export并不能帶來像普通函數那樣的真正的分離式編譯。

            準則記住,export只能隱藏依賴性,并不能消除依賴性。

            準則(就目前而言)避免使用export。

             

            第十一條 第十二條 異常安全問題

            準則

            • 為應用程序或子系統確立一個整體的錯誤報告/處理策略,并始終遵循它。并為錯誤報告、錯誤傳播以及錯誤處理制定策略。
            • 在那些偵測到錯誤而自身又無法對其進行處理的地方拋出異常。
            • 在那些具備有足夠知識和上下文信息去處理/轉換錯誤或強制實施邊界條件的地方編寫try/catch(例如,在子系統邊界上或其他運行時防火墻邊界上進行catch(…))。

            準則函數應當總是支持它所能支持的最強的異常安全保證,但前提是不能給那些并不需要該保證的調用者帶來額外開銷。

            準則永遠不要允許析構函數、釋放操作以及swap()函數拋出任何異常,因為否則的話,就沒法安全且可靠地進行資源清理了。

            準則

            • 永遠不要為函數加上異常規格。
            • 除非你想聲明的是空異常規格列表(即throw()),但仍不推薦這樣做。
            午夜精品久久久久9999高清| 91精品国产乱码久久久久久| 亚洲国产成人精品91久久久| 久久综合给合久久国产免费 | 精品国产一区二区三区久久蜜臀| 久久天天躁狠狠躁夜夜96流白浆| 亚洲精品乱码久久久久久自慰| 亚洲av伊人久久综合密臀性色| 伊人久久一区二区三区无码| 国产巨作麻豆欧美亚洲综合久久| 国内精品久久久久久久久| 久久996热精品xxxx| 欧美精品丝袜久久久中文字幕 | 久久综合狠狠综合久久激情 | 久久这里只有精品首页| 久久亚洲国产成人影院网站| 97久久超碰国产精品2021| 国産精品久久久久久久| 久久天天躁狠狠躁夜夜av浪潮 | 国产一区二区久久久| 精品人妻久久久久久888| 99久久国产综合精品五月天喷水| 久久免费香蕉视频| 国产精品久久久久久久| 性欧美大战久久久久久久| 99久久99久久精品免费看蜜桃| 久久综合伊人77777麻豆| 久久九九精品99国产精品| 人人狠狠综合88综合久久| 久久婷婷激情综合色综合俺也去| a级毛片无码兔费真人久久| 国产偷久久久精品专区| 狠狠色婷婷综合天天久久丁香| 久久久久亚洲AV成人网人人网站| 狠色狠色狠狠色综合久久| 久久久精品人妻一区二区三区四 | 国产香蕉97碰碰久久人人| 69国产成人综合久久精品| 少妇久久久久久久久久| 久久久久久无码国产精品中文字幕 | 99精品国产在热久久无毒不卡|