• <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>

            C小加

            厚德 博學 求真 至善 The bright moon and breeze
            posts - 145, comments - 195, trackbacks - 0, articles - 0
              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

            避免使用宏

            Posted on 2011-11-19 13:52 C小加 閱讀(13328) 評論(6)  編輯 收藏 引用

            概述:

                    宏是C和C++語言的抽象設施中最生硬的工具,它是披著函數外衣的饑餓的狼,很難馴服,它會我行我素地游走于各處。要避免使用宏。

            討論:

                    在C++中,幾乎從不需要使用宏。

                    可以用const或者enum定義易于理解的常量,用inline避免函數調用的開銷,用template指定函數系列和類型系列,用namespace避免名稱沖突。

                C++的宏的主要問題在于,它們表面上看起來很好,而實際上做的卻是另一回事。宏會忽略作用域,忽略類型系統,忽略所有其他的語言特性和規則,而且會劫持它為文件其余部分所定義(#define)的符號。宏調用看上去很像符號或者函數調用,但實際上并非如此。宏不太“衛生”,也就是說,它會根據自己被使用時所處的環境引人注目而且令人驚奇地展開為各種東西。宏需要進行文本替換,因此編寫遠距離也正確的宏接近于一種魔法,而精通這種魔法既無意義又無趣味。

                不少人認為與模板相關的錯誤都是最難以解讀的,他們可能還沒有看到誤寫和誤用的宏所引起的那些錯誤。模板是C++類型系統的一部分,因此編譯器可以更好地對它們進行處理,而宏天生是與語言本身割裂開來的,因此很難處理。更糟的是,與模板不同,宏可能展開為在偶然情況下能夠編譯的“傳輸線噪音”。最后,宏中的錯誤可能只有在宏展開之后才能被報告出來,而不是在定義時。

                 即使在極少的情況下,有正當理由編寫宏,也決不要考慮編寫一個以常見詞或者縮略語為名字的宏。盡可能快的取消宏的定義(#undef)。

            示例:

                1、定義一個宏#define  min(n, m)    ((n) < (m) ? (n) : (m)) 

                        定義兩個變量a和b,min(++a, b) 傳入之后是這樣 ((++a) < (b) ? (++a) : (b))   如果++a小于b的話,a就自加了兩次,很明顯不符合宏使用的初衷。

                2、將模板實例化轉給宏,宏僅能理解C語言的小括號和方括號,并將其進行匹配。然而,C++又定義了一個新的括號結構,即模板中使用的尖括號<和>。宏無法正確的匹配它們,這意味著在下面的宏調用中:

                   MACRO(Foo<int,double>)

                   宏會認為傳給自己的是兩個參數,即Foo<int和double>,而事實上該結構是一個C++實體。

            例外情況:

                宏仍然是幾個重要任務的唯一解決方案,比如#include保護符,條件編譯中的#ifdef和#ifndef,以及assert的實現。

                在條件編譯中,要避免在代碼中到處雜亂地插入#ifdef。相反,應該對代碼進行組織,利用宏在驅動一個公共接口的多個實現,然后始終使用該接口。

                如果不想到處復制粘貼代碼段,那么可以使用宏,但要非常小心。
                   


                  來自:《C++編程規范》

            Feedback

            # re: 避免使用宏  回復  更多評論   

            2011-11-21 09:13 by yanxinmeng
            最近寫的代碼 涉及到一些別人寫的宏。 其中還有 臨時變量。。。

            # re: 避免使用宏  回復  更多評論   

            2011-11-21 16:04 by C小加
            使用的時候要注意,有時候會得到莫名其妙的答案,你還不知道該怎么改@yanxinmeng

            # re: 避免使用宏[未登錄]  回復  更多評論   

            2011-12-07 08:48 by alex
            轉載請注明出處。

            這段話 引自 《C++編程規范101條規則、準則最佳實踐》 書里面的第16條: 避免使用宏。

            很不認同,轉載不標明出處的。不尊重原作者,對讀者也不負責。

            # re: 避免使用宏  回復  更多評論   

            2012-03-04 12:04 by 泡菜
            規范是死的,人是活的!!

            舉個例子;不少書/規范主張不用goto...有空下個Linux內核代碼看看,goto遍地都是!Linux內核就是依靠goto一點一點把性能給壓榨出來的(至于是如何壓榨的自己想想,俺不解釋)

            不少主張不用goto的,有幾塊能達到寫內核的水平??

            # re: 避免使用宏  回復  更多評論   

            2012-03-04 13:28 by C小加
            聽人說搞嵌入式的用的不是循環,是goto,呵呵。自從老師說不要用goto,我就從來都沒用過。看來goto是提高效率的利器啊,可是危險性就不用考慮么?疑惑@泡菜

            # re: 避免使用宏  回復  更多評論   

            2012-03-15 18:34 by coreBugZJ
            個人覺得,各種機制都是有其存在的價值,不必撟枉過正。

            # re: 避免使用宏  回復  更多評論   

            2012-03-16 10:09 by C小加
            贊同。宏在嵌入式中宏也占很大比重。這只是對一些初學者來說的,使用宏的時候一定要萬分小心。
            @coreBugZJ
            久久午夜综合久久| 国产精品久久久久aaaa| 久久97久久97精品免视看秋霞| 99国产欧美精品久久久蜜芽| 99久久成人国产精品免费| 久久精品免费观看| 久久久久亚洲精品日久生情| 久久人人爽人人爽人人AV东京热 | 久久精品国产99国产精偷| 观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 久久精品人成免费| 久久国产精品波多野结衣AV| 久久人人爽人人爽人人片AV不| 国产产无码乱码精品久久鸭| 久久午夜综合久久| 国产精品岛国久久久久| 精品国产日韩久久亚洲| 国内精品久久久久久久coent| 久久精品国产男包| 久久久久久亚洲精品无码| 久久久精品人妻一区二区三区蜜桃 | 久久亚洲高清综合| 99精品国产在热久久无毒不卡| 久久亚洲国产成人精品无码区| 久久亚洲欧美日本精品| 久久精品国产精品亚洲毛片| 亚洲欧美伊人久久综合一区二区 | 亚洲国产精品一区二区久久| 久久久久久国产精品免费无码 | 久久综合九色综合久99| 亚洲AV日韩精品久久久久久久| 久久精品国产亚洲精品| 2021国产成人精品久久| 国产午夜免费高清久久影院| 无码人妻久久一区二区三区免费丨| 中文字幕精品无码久久久久久3D日动漫| 91久久精品视频| 国产AⅤ精品一区二区三区久久 | 久久无码人妻精品一区二区三区 | 日产精品久久久久久久| 久久天天躁狠狠躁夜夜网站|