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

            從一道面試題來闡釋一個普遍的認知誤區(qū)

                                                                                                              peakflys原創(chuàng)作品,轉(zhuǎn)載請保留原作者和源鏈接
                上午一個師弟在QQ上問我一道筆試題,是他前兩天去KONAMI面試時做的,這道題大致是這樣的:
                    解釋以下語句的含義:
                     1、new A;
                     2、new A();   
             
               也許很多人包括我自己,都可以馬上給出第一種情況的答案:在堆上為A類分配內(nèi)存,然后調(diào)用A的構(gòu)造函數(shù)。這種說法被大家所熟知,因為包括《STL源碼剖析》等大作在內(nèi)也都是這么寫的(但是你認為這種說法完全正確嗎?其實不盡然,答案后面揭曉)
                第二種情況,對象構(gòu)造的時候初始化列表為空會和第一種有什么不同呢?對于這種在實際工程中很少使用的情況,我一時還真給不出確切的答案。
               網(wǎng)上搜了一下,看到CSDN里面還有專門針對這個問題的一個帖子(原帖鏈接 http://bbs.csdn.net/topics/320161716)。
               好像最終也沒有可以信服的答案,認同度比較高的是這樣的說法:“加括號調(diào)用沒有參數(shù)的構(gòu)造函數(shù),不加括號調(diào)用默認構(gòu)造函數(shù)或唯一的構(gòu)造函數(shù),看需求” (peakflys注:這種說法是錯誤的,答案后面揭曉)
               既然沒有特別靠譜的答案,不如自己動手找出答案。
               構(gòu)造以下示例:
            /**
             *\brief example1 difference between new and new()
             *\author peakflys
             *\data 12:10:24 Monday, April 08, 2013
             
            */

            class A
            {
            public:
                int a;
            };

            int main()
            {
                A *pa = new A;
                A *paa = new A();
                return 0;
            }
            查看main函數(shù)的匯編代碼(編譯器:gcc (GCC) 4.4.6 20120305 (Red Hat 4.4.6-4) )
            int main()
            {
              4005c4:   55                      push   %rbp
              4005c5:   48 89 e5                mov    %rsp,%rbp
              4005c8:   48 83 ec 10             sub    $0x10,%rsp
                A *pa = new A;
              4005cc:   bf 04 00 00 00          mov    $0x4,%edi
              4005d1:   e8 f2 fe ff ff          callq  4004c8 <_Znwm@plt>         //調(diào)用new
              4005d6:   48 89 45 f0             mov    %rax,-0x10(%rbp)           //rax寄存器內(nèi)容賦給指針pa(rax寄存器里是new調(diào)用產(chǎn)生的A對象堆內(nèi)存地址)
                A *paa = new A();
              4005da:   bf 04 00 00 00          mov    $0x4,%edi
              4005df:   e8 e4 fe ff ff          callq  4004c8 <_Znwm@plt>         //調(diào)用new
              4005e4:   48 89 c2                mov    %rax,%rdx                      //rax的內(nèi)容放入rdx,執(zhí)行之后,rdx里存放的即是通過new A()產(chǎn)生的內(nèi)存地址
              4005e7:   c7 02 00 00 00 00       movl   $0x0,(%rdx)                 //把rdx內(nèi)存指向的內(nèi)容賦為0值,即把A::a賦值為0
              4005ed:   48 89 45 f8             mov    %rax,-0x8(%rbp)             //rax寄存器內(nèi)容賦給指針paa(rax寄存器里是new()調(diào)用產(chǎn)生的A對象堆內(nèi)存地址)
                 return 0;
              4005f1:   b8 00 00 00 00          mov    $0x0,%eax
            }
              4005f6:   c9                      leaveq 
              4005f7:   c3                      retq
                通過上面產(chǎn)生的匯編代碼(對AT&T匯編不熟悉的可以看注釋)可以很容易看出,new A()的執(zhí)行,在調(diào)用完operator new分配內(nèi)存后,馬上對新分配內(nèi)存中的對象使用0值初始化,而new A 僅僅是調(diào)用了operator new分配內(nèi)存!
               是不是這樣就可以下結(jié)論 new A()比new A多了一步,即初始化對象的步驟呢?
               我們再看看下面這種情況:
            /**
             *\brief example2 difference between new and new()
             *\author peakflys
             *\data 12:23:20 Monday, April 08, 2013
             
            */

            class A
            {
            public:
                A(){a = 10;}
                int a;
            };

            int main()
            {
                A *pa = new A;
                A *paa = new A();
                return 0;
            }
               這種情況是類顯示提供含默認值的構(gòu)造函數(shù)。
               查看匯編實現(xiàn)如下:
            int main()
            {
              4005c4:   55                      push   %rbp
              4005c5:   48 89 e5                mov    %rsp,%rbp
              4005c8:   53                      push   %rbx
              4005c9:   48 83 ec 18             sub    $0x18,%rsp
                A *pa = new A;
              4005cd:   bf 04 00 00 00          mov    $0x4,%edi
              4005d2:   e8 f1 fe ff ff          callq  4004c8 <_Znwm@plt>
              4005d7:   48 89 c3                mov    %rax,%rbx
              4005da:   48 89 d8                mov    %rbx,%rax
              4005dd:   48 89 c7                mov    %rax,%rdi
              4005e0:   e8 2d 00 00 00          callq  400612 <_ZN1AC1Ev>
              4005e5:   48 89 5d e0             mov    %rbx,-0x20(%rbp)
                A *paa = new A();
              4005e9:   bf 04 00 00 00          mov    $0x4,%edi
              4005ee:   e8 d5 fe ff ff          callq  4004c8 <_Znwm@plt>
              4005f3:   48 89 c3                mov    %rax,%rbx
              4005f6:   48 89 d8                mov    %rbx,%rax
              4005f9:   48 89 c7                mov    %rax,%rdi
              4005fc:   e8 11 00 00 00          callq  400612 <_ZN1AC1Ev>
              400601:   48 89 5d e8             mov    %rbx,-0x18(%rbp)
                return 0;
              400605:   b8 00 00 00 00          mov    $0x0,%eax
            }
              40060a:   48 83 c4 18             add    $0x18,%rsp
              40060e:   5b                      pop    %rbx
              40060f:   c9                      leaveq 
              400610:   c3                      retq 
               上面的匯編代碼就不在添加注釋了,因為兩種操作產(chǎn)生的匯編代碼是一樣的,都是先調(diào)用operator new分配內(nèi)存,然后調(diào)用構(gòu)造函數(shù)。
               上面的情況在VS2010下驗證是一樣的情況,有興趣的朋友可以自己去看,這里就不再貼出VS2010下的匯編代碼了。
               通過上面的分析,對于new A和 new A() 的區(qū)別,我們可以得出下面的結(jié)論:
                  1、類體含有顯示適合地默認構(gòu)造函數(shù)時,new A和new A()的作用一致,都是首先調(diào)用operator new分配內(nèi)存,然后調(diào)用默認構(gòu)造函數(shù)初始化對象。
                  2、類體無顯示構(gòu)造函數(shù)時,new A()首先調(diào)用operator new來為對象分配內(nèi)存,然后使用空值初始化對象成員變量,而new A僅僅是調(diào)用operator new分配內(nèi)存,對象的成員變量是無意義的隨機值!  (peakflys注:對于基本數(shù)據(jù)類型,如int等 適用此條)
               注意到,現(xiàn)在很多書籍對new操作符的說明都存在紕漏,例如《STL源碼剖析》中2.2.2節(jié)中有以下的描述:

            事實證明,new Foo的操作是否有構(gòu)造函數(shù)的調(diào)用是不確定的,具體要看Foo類體里是否有顯示構(gòu)造函數(shù)的出現(xiàn)。

                                                                                                            by peakflys 13:40:00 Monday, April 08, 2013

            /*****************************************華麗分割線**************************************
            補充:剛才發(fā)現(xiàn),在C++Primer第四版5.11節(jié)中,已經(jīng)有了對于new A()的說明:
               We indicate that we want to value-initialize the newly allocated object by following the type nameby a pair of empty parentheses. The empty parentheses signal that we want initialization but arenot supplying a specific initial value. In the case of class types (such as string) that define their own constructors, requesting value-initialization is of no consequence: The object is initialized by running the default constructor whether we leave it apparently uninitialized orask for value-initialization. In the case of built-in types or types that do not define any constructors, the difference is significant:
                 int *pi = new int;         // pi points to an uninitialized int 
                 int *pi = new int();       // pi points to an int value-initialized to 0 
            In the first case, the int is uninitialized; in the second case, the int is initialized to zero.
               這里給出的解釋和上面自己分析的new A()的行為是一致的。
            /***************************************再次華麗分割線************************************
            鑒于上面的結(jié)論是通過GCC和VS2010得出的,而且有朋友也提出同樣的質(zhì)疑,為了確定這種結(jié)果是否是編譯器相關(guān)的,剛才特意查看了一下C++的標準化文檔。
            摘自:ISO/IEC 14882:2003(E) 5.3.4 - 15
            — If the new-initializer is omitted:
                  — If T is a (possibly cv-qualified) non-POD class type (or array thereof), the object is default-initialized(8.5). If T is a const-qualified type, the underlying class type shall have a user-declared default constructor.
                  — Otherwise, the object created has indeterminate value. If T is a const-qualified type, or a (possibly cv-qualified) POD class type (or array thereof) containing (directly or indirectly) a member of const-qualified type, the program is ill-formed;
            — If the new-initializer is of the form (), the item is value-initialized (8.5);
            所以可以確定,這種情況完全是編譯器無關(guān)的(當然那些不完全按照標準實現(xiàn)的編譯器除外)。
            但是通過上面標準化文檔的描述,我們可以看出文中對new A在無顯示構(gòu)造函數(shù)時的總結(jié)并不是特別全面,鑒于很多公司都有這道面試題(撇去這些題目的實際考察意義不說),我們有必要再補充一下:   對于new A: 這樣的語句,再調(diào)用完operator new分配內(nèi)存之后,如果A類體內(nèi)含有POD類型,則POD類型的成員變量處于未定義狀態(tài),如果含有非POD類型則調(diào)用該類型的默認構(gòu)造函數(shù)。而 new A()在這些情況下都會初始化。
               PS:估計很多公司的“正確答案“ 也不一定正確吧。

            posted on 2013-04-08 13:43 peakflys 閱讀(6827) 評論(21)  編輯 收藏 引用 所屬分類: C++

            評論

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-04-08 18:09 Richard Wei

            分析的不錯,支持下.  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-04-08 18:34 zgpxgame

            分析的不錯,但就這道題目來說,感覺這題考的意義不大,面試過程用這種細枝末節(jié)的語法題考察不出面試者的思維。  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-04-09 01:29 Wilbur

            Default initialization和value initialiaztion,http://en.cppreference.com/w/cpp/language/default_initialization
            http://en.cppreference.com/w/cpp/language/value_initialization  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-04-09 09:51 zuhd

            int *pi = new int; // pi points to an uninitialized int
            int *pi = new int(); // pi points to an int value-initialized to 0
            這樣也行?
            AT&T的匯編看的好別扭啊  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-04-09 09:54 tangfu

            http://en.cppreference.com/w/cpp/language/default_initialization語言標準提到了幾點

            1. when an object with dynamic storage duration is created by a new-expression without an initializer
            2. If T is a class type, the default constructor is called to provide the initial value for the new object.

            感覺你這個是屬于語言灰色地帶,不同編譯器的實現(xiàn)有關(guān)系啊。。。。  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-04-09 10:17 peakflys

            其實現(xiàn)在國內(nèi)很多公司的筆試題都很學究,真正考察在工程項目中使用的比較多的那些語法、算法或者一些編程技巧很少、很淺……@zgpxgame
              回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-04-09 10:22 peakflys

            恩,這個網(wǎng)站提供了很好的C++參考文檔@Wilbur
              回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-04-09 10:58 peakflys

            @tangfu
            其實這個不是編譯器相關(guān)的,標準化委員會細化規(guī)則的程度大大超過你的想象。
            摘自:ISO/IEC 14882:2003(E) 5.3.4 - 15
            — If the new-initializer is omitted:
            — If T is a (possibly cv-qualified) non-POD class type (or array thereof), the object is default-initialized(8.5). If T is a const-qualified type, the underlying class type shall have a user-declared default constructor.
            — Otherwise, the object created has indeterminate value. If T is a const-qualified type, or a (possibly cv-qualified) POD class type (or array thereof) containing (directly or indirectly) a member of const-qualified type, the program is ill-formed;
            — If the new-initializer is of the form (), the item is value-initialized (8.5);  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-04-09 11:20 溪流

            @peakflys
            學習了
              回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū)[未登錄] 2013-04-14 09:50 peakflys

            看多了就很自然了@zuhd
              回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-04-14 09:57 天馬星空

            @peakflys
            學習了,我們公司筆試題也有這一題,剛才看了一下標準答案,確實沒有像樓主分析的這么細,只要寫出初始化的區(qū)別就行了。  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū)[未登錄] 2013-05-03 11:21 noname

            你這完全是因為debug才有的問題, vs2012 release證明,所謂的初始化代碼不會產(chǎn)生,其它版本應該也是一樣的  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū)[未登錄] 2013-05-03 11:24 noname

            class A{
            int a;
            };
            #pragma optimize=2

            int _tmain(int argc, _TCHAR* argv[])
            {
            auto x = new A;
            x = new A();

            return 0;
            }




            auto x = new A;
            003A1000 push 4
            003A1002 call dword ptr ds:[3A2090h]
            x = new A();
            003A1008 push 4
            003A100A call dword ptr ds:[3A2090h]
            003A1010 add esp,8
            003A1013 test eax,eax
            003A1015 je wmain+1Dh (03A101Dh)
            003A1017 mov dword ptr [eax],0

            return 0;
            003A101D xor eax,eax
            }
            003A101F ret   回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū)[未登錄] 2013-05-10 09:25 路人甲

            2012產(chǎn)生的匯編這么奇怪?不過如果上面匯編是release版本的話,不正好和樓主的觀點一樣嗎? 看 003A1017,明顯的0賦值操作啊。搬起石頭砸自己的腳了,哈哈 @noname
              回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-05-10 14:30 jxhgzs

            我編程的時候從來沒有使用過 new A 這種形式,都是 new A() ;但是也經(jīng)常碰到數(shù)據(jù)取隨機數(shù)值的問題,我得到的解釋是 類成員變量不會賦初始值,如果是靜態(tài)的 基本類型 會有初始值,求解!  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-05-10 17:29 syd

            由編譯器構(gòu)造出來的構(gòu)造函數(shù)是trivial的,不會對member初始為0的  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-05-10 17:52 peakflys

            @noname
            文中的結(jié)論是基于C++98標準的編譯器驗證結(jié)果,而VS2012是根據(jù)最新的C++11標準實現(xiàn)的,新標準對于這種情況的規(guī)定有沒有變動尚未驗證。
            PS:你提供的匯編代碼應該是和文中的結(jié)論吻合的吧?  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-05-10 17:54 peakflys

            @jxhgzs
            請?zhí)峁┠愕木幾g器版本,因為這是C++98標準規(guī)定的操作,如果你的編譯器產(chǎn)生的代碼行為同文中描述有出入,只能說明你的編譯器沒按照98標準來實現(xiàn)。  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-05-10 17:58 peakflys

            @syd
            其實不然,編譯器產(chǎn)生的構(gòu)造函數(shù)很多時候是nontrivial的,
            例如:成員變量是含有構(gòu)造函數(shù)的類類型,含有virtual function等等  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2013-12-12 15:18 挑燈看劍

            表達式 POD類型T non-POD類型T
            new T 不初始化 缺省初始化
            new T() 總是缺省初始化  回復  更多評論   

            # re: 從一道面試題來闡釋一個普遍的認知誤區(qū) 2014-04-02 14:06 allen

            贊 ! 態(tài)度很認真 !  回復  更多評論   

            <2025年5月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            導航

            統(tǒng)計

            公告

            人不淡定的時候,就愛表現(xiàn)出來,敲代碼如此,偶爾的靈感亦如此……

            常用鏈接

            留言簿(4)

            隨筆分類

            隨筆檔案

            文章檔案

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            26uuu久久五月天| 中文精品久久久久国产网址| 久久天天躁夜夜躁狠狠| 欧美亚洲国产精品久久| 久久午夜伦鲁片免费无码| 日本久久久久久中文字幕| 日韩美女18网站久久精品| 男女久久久国产一区二区三区| 国产精品久久久久久一区二区三区| 精品久久久久中文字幕一区| 97精品伊人久久大香线蕉| 成人午夜精品久久久久久久小说| 亚洲七七久久精品中文国产| 青青青青久久精品国产h| 久久人人爽人人爽人人片av麻烦| 日本三级久久网| 国内精品久久久久影院优| 伊人久久成人成综合网222| 精品久久久久久中文字幕| 无码任你躁久久久久久老妇App| 2021国产成人精品久久| 精品乱码久久久久久久| 亚洲香蕉网久久综合影视| 合区精品久久久中文字幕一区| www.久久热.com| 久久久久亚洲AV无码专区体验| 人妻中文久久久久| 久久久久久久国产免费看| 久久香蕉一级毛片| 精品久久久久久国产91| 国产精品久久99| 99久久人妻无码精品系列蜜桃 | 亚洲国产精品无码久久久蜜芽| 国产精品欧美亚洲韩国日本久久 | 久久国产精品免费一区| 国产一区二区三区久久| 久久精品欧美日韩精品| 久久青青草原亚洲av无码app| 欧洲人妻丰满av无码久久不卡| 亚洲国产精品久久电影欧美| 久久精品麻豆日日躁夜夜躁|