青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

面對現實,超越自己
逆水行舟,不進則退
posts - 269,comments - 32,trackbacks - 0

   解釋以下語句的含義:
         1
new A;
         2
new A();    
   也許很多人包括我自己,都可以馬上給出第一種情況的答案:在堆上為A類分配內存,然后調用A的構造函數。這種說法被大家所熟知,因為包括《STL源碼剖析》等大作在內也都是這么寫的(但是你認為這種說法完全正確嗎?
其實不盡然,答案后面揭曉)
    第二種情況,對象構造的時候初始化列表為空會和第一種有什么不同呢?對于這種在實際工程中很少使用的情況,我一時還真給不出確切的答案。

   網上搜了一下,看到CSDN里面還有專門針對這個問題的一個帖子(原帖鏈接 http://bbs.csdn.net/topics/320161716)。

   好像最終也沒有可以信服的答案,認同度比較高的是這樣的說法:
加括號調用沒有參數的構造函數,不加括號調用默認構造函數或唯一的構造函數,看需求peakflys注:這種說法是錯誤的,答案后面揭曉)
   既然沒有特別靠譜的答案,不如自己動手找出答案。

   構造以下示例:

/**
 *\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函數的匯編代碼(編譯器: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>         //調用new
  4005d6:   48 89 45 f0             mov    %rax,-0x10(%rbp)           //rax寄存器內容賦給指針pa(rax寄存器里是new調用產生的A對象堆內存地址)
    A *paa = new A();
  4005da:   bf 04 00 00 00          mov    $0x4,%edi
  4005df:   e8 e4 fe ff ff          callq  4004c8 <_Znwm@plt>         //調用new
  4005e4:   48 89 c2                mov    %rax,%rdx                      //rax的內容放入rdx,執行之后,rdx里存放的即是通過new A()產生的內存地址
  4005e7:   c7 02 00 00 00 00       movl   $0x0,(%rdx)                 //把rdx內存指向的內容賦為0值,即把A::a賦值為0
  4005ed:   48 89 45 f8             mov    %rax,-0x8(%rbp)             //rax寄存器內容賦給指針paa(rax寄存器里是new()調用產生的A對象堆內存地址)
     return 0;
  4005f1:   b8 00 00 00 00          mov    $0x0,%eax
}
  4005f6:   c9                      leaveq 
  4005f7:   c3                      retq
    通過上面產生的匯編代碼(AT&T匯編不熟悉的可以看注釋)可以很容易看出,new A()的執行,在調用完operator new分配內存后,馬上對新分配內存中的對象使用0值初始化,而new A 僅僅是調用了operator new分配內存!
   是不是這樣就可以下結論 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;
}

   這種情況是類顯示提供含默認值的構造函數。
   查看匯編實現如下:

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 

   上面的匯編代碼就不在添加注釋了,因為兩種操作產生的匯編代碼是一樣的,都是先調用operator new分配內存,然后調用構造函數。
   上面的情況在VS2010下驗證是一樣的情況,有興趣的朋友可以自己去看,這里就不再貼出VS2010下的匯編代碼了。

   通過上面的分析,對于
new A new A() 的區別,我們可以得出下面的結論:
      
1、類體含有顯示適合地默認構造函數時,new Anew A()的作用一致,都是首先調用operator new分配內存,然后調用默認構造函數初始化對象。
     2、類體無顯示構造函數時,new A()首先調用operator new來為對象分配內存,然后使用空值初始化對象成員變量,而new A僅僅是調用operator new分配內存,對象的成員變量是無意義的隨機值!  peakflys注:對于基本數據類型,如int 適用此條)
   注意到,現在很多書籍對new操作符的說明都存在紕漏,例如《STL源碼剖析》中2.2.2節中有以下的描述:


事實證明,new Foo的操作是否有構造函數的調用是不確定的,具體要看Foo類體里是否有顯示構造函數的出現。

/*****************************************華麗分割線**************************************
補充:剛才發現,在C++Primer第四版5.11節中,已經有了對于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()的行為是一致的。

/***************************************再次華麗分割線************************************
鑒于上面的結論是通過GCCVS2010得出的,而且有朋友也提出同樣的質疑,為了確定這種結果是否是編譯器相關的,剛才特意查看了一下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);

所以可以確定,這種情況完全是編譯器無關的(當然那些不完全按照標準實現的編譯器除外)
但是通過上面標準化文檔的描述,我們可以看出文中對new A在無顯示構造函數時的總結并不是特別準確,鑒于很多公司都有這道面試題(撇去這些題目的實際考察意義不說),我們有必要再補充一下: 對于new A: 這樣的語句,再調用完operator new分配內存之后,如果A類體內含有POD類型,則POD類型的成員變量處于未定義狀態,如果含有非POD類型則調用該類型的默認構造函數。而 new A()在這些情況下都會初始化。
   PS:估計很多公司的正確答案“ 也不一定正確吧。

 

本文轉自:http://m.shnenglu.com/peakflys/archive/2013/04/08/199208.html

 

posted on 2013-04-18 11:03 王海光 閱讀(488) 評論(0)  編輯 收藏 引用 所屬分類: 其他
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美日韩影院| 亚洲第一精品影视| 在线观看中文字幕亚洲| 国产精品一区二区三区久久久| 欧美日本中文| 欧美午夜片欧美片在线观看| 欧美三日本三级少妇三2023| 国产精品久久久久久超碰| 国产农村妇女精品一区二区| 国外成人在线视频| 亚洲精品中文字幕在线| 中文一区二区在线观看| 欧美一级欧美一级在线播放| 亚洲最新视频在线| 欧美制服丝袜| 久久免费高清| 久久精品国产亚洲精品| 日韩视频在线观看国产| 亚洲激情午夜| 老司机久久99久久精品播放免费 | 亚洲激情一区二区| 亚洲免费电影在线| 欧美一区二区播放| 免费成人av在线看| 国产精品免费久久久久久| 国产日韩亚洲欧美精品| 亚洲国产毛片完整版| 亚洲视频一区在线| 久久先锋影音av| 亚洲狠狠婷婷| 久久本道综合色狠狠五月| 欧美激情精品久久久| 国产欧美欧美| 99热精品在线| 欧美本精品男人aⅴ天堂| 亚洲视频一区在线| 欧美成年视频| 一区二区视频在线观看| 亚洲小视频在线观看| 欧美va天堂va视频va在线| 亚洲一区二区三区色| 欧美精品v国产精品v日韩精品| 国产丝袜一区二区三区| 在线综合亚洲| 91久久精品www人人做人人爽| 久久国产精品色婷婷| 国产精品久久久久久久久久ktv | 麻豆成人综合网| 国产久一道中文一区| 亚洲私人黄色宅男| 亚洲精品久久嫩草网站秘色| 欧美在线首页| 国产网站欧美日韩免费精品在线观看| 亚洲天堂偷拍| 亚洲最新在线视频| 欧美日韩三区| 中日韩美女免费视频网站在线观看 | 美女脱光内衣内裤视频久久影院 | 久久久一本精品99久久精品66| 欧美日韩一区免费| 亚洲国产黄色| 免费在线欧美视频| 久久久久久香蕉网| 伊人成人在线视频| 裸体歌舞表演一区二区| 久久久精品日韩欧美| 激情欧美日韩一区| 免费在线看一区| 免费久久精品视频| 99天天综合性| 中文亚洲欧美| 国产精品久久看| 久久精品官网| 欧美一区激情| 亚洲高清色综合| 亚洲精品一区二区三区av| 欧美日韩精品国产| 午夜精品福利一区二区蜜股av| 亚洲伊人一本大道中文字幕| 国产一区二区三区丝袜| 欧美大片第1页| 欧美日本在线视频| 午夜视频在线观看一区二区| 欧美一站二站| 亚洲精品黄色| 亚洲一区二区三区激情| 黄色免费成人| 亚洲精品日产精品乱码不卡| 国产精品日韩高清| 免费不卡亚洲欧美| 欧美日韩喷水| 久久夜色撩人精品| 欧美日韩第一区日日骚| 午夜精品婷婷| 欧美99在线视频观看| 亚洲欧美一区二区激情| 久久深夜福利免费观看| 亚洲午夜激情在线| 久久一本综合频道| 亚洲欧美日本视频在线观看| 久久久久看片| 亚洲欧洲99久久| 女生裸体视频一区二区三区| 欧美一级久久| 欧美激情视频在线播放| 久久久99国产精品免费| 欧美日韩高清在线播放| 美女主播一区| 国产日韩精品在线播放| 亚洲精品乱码久久久久久按摩观 | 国产在线观看精品一区二区三区| 亚洲黄色大片| 国产一区二区福利| 一区二区三区高清视频在线观看| 在线观看欧美激情| 午夜精品国产| 亚洲日韩视频| 毛片基地黄久久久久久天堂| 亚洲二区三区四区| 夜夜夜久久久| 亚洲激情图片小说视频| 久久av二区| 欧美一区二区精品久久911| 欧美精品在线一区二区三区| 免费在线亚洲欧美| 一区二区三区在线观看欧美| 午夜精品免费| 欧美在线播放一区| 国产精品毛片高清在线完整版| 亚洲三级电影在线观看| 亚洲三级国产| 欧美精品一区二区三区一线天视频| 欧美freesex交免费视频| 国产欧美精品va在线观看| 一区二区三区免费观看| 亚洲婷婷国产精品电影人久久| 欧美激情按摩| 亚洲国产精品精华液网站| 亚洲国产中文字幕在线观看| 裸体歌舞表演一区二区| 欧美1区免费| 亚洲欧洲日产国产网站| 免费亚洲视频| 亚洲人成77777在线观看网| 99热免费精品| 欧美午夜精品久久久久免费视| 一区二区三区国产盗摄| 欧美一区二区在线观看| 国产一区二区三区四区五区美女| 午夜精品影院| 欧美不卡视频一区| 99re热这里只有精品免费视频| 欧美美女操人视频| 一区二区三区你懂的| 性色av一区二区怡红| 国产亚洲欧美日韩在线一区| 久久久久久久一区二区| 亚洲国产福利在线| 亚洲视频二区| 国产精品综合av一区二区国产馆| 欧美一区二区三区另类| 欧美电影资源| 亚洲一区免费观看| 国产一区日韩欧美| 欧美国产一区视频在线观看 | 久久久99爱| 亚洲激情校园春色| 欧美一级午夜免费电影| 精品成人一区二区| 欧美精品一区在线| 校园激情久久| 91久久综合| 久久精品男女| 日韩视频亚洲视频| 国产欧美精品日韩精品| 你懂的国产精品| 亚洲欧美精品在线观看| 欧美国产一区二区在线观看 | 99re国产精品| 久热精品视频在线观看一区| 艳妇臀荡乳欲伦亚洲一区| 国产手机视频一区二区| 免费亚洲电影在线观看| 欧美中文在线观看| 蜜月aⅴ免费一区二区三区| 亚洲精品国久久99热| 国产欧美一区二区视频| 欧美国产日韩一区二区| 欧美在线观看一区| 99精品久久| 欧美激情一区二区三区四区| 欧美一区二区免费观在线| 亚洲精品美女久久7777777| 国产精品揄拍500视频| 欧美精品二区| 麻豆精品在线观看| 久久国产乱子精品免费女| 亚洲一区二区三区精品动漫| 亚洲毛片一区| 亚洲国产日韩精品|