<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          C++ 初始化列表

          共 696字,需瀏覽 2分鐘

           ·

          2022-03-01 22:51

          作者:翰墨小生

          鏈接:https://www.cnblogs.com/graphics/archive/2010/07/04/1770900.html

          何謂初始化列表

          與其他函數(shù)不同,構(gòu)造函數(shù)除了有名字,參數(shù)列表和函數(shù)體之外,還可以有初始化列表,初始化列表以冒號(hào)開(kāi)頭,后跟一系列以逗號(hào)分隔的初始化字段。在C++中,struct和class的唯一區(qū)別是默認(rèn)的訪問(wèn)性不同,而這里我們不考慮訪問(wèn)性的問(wèn)題,所以下面的代碼都以struct來(lái)演示。


          struct?foo
          {

          ????string?name?;
          ????int?id?;
          ????foo(string?s,?int?i):name(s),?id(i){}?;?//?初始化列表
          };

          構(gòu)造函數(shù)的兩個(gè)執(zhí)行階段

          構(gòu)造函數(shù)的執(zhí)行可以分成兩個(gè)階段,初始化階段和計(jì)算階段,初始化階段先于計(jì)算階段。


          初始化階段


          所有類類型(class type)的成員都會(huì)在初始化階段初始化,即使該成員沒(méi)有出現(xiàn)在構(gòu)造函數(shù)的初始化列表中。


          計(jì)算階段


          一般用于執(zhí)行構(gòu)造函數(shù)體內(nèi)的賦值操作,下面的代碼定義兩個(gè)結(jié)構(gòu)體,其中Test1有構(gòu)造函數(shù),拷貝構(gòu)造函數(shù)及賦值運(yùn)算符,為的是方便查看結(jié)果。Test2是個(gè)測(cè)試類,它以Test1的對(duì)象為成員,我們看一下Test2的構(gòu)造函數(shù)是怎么樣執(zhí)行的。


          struct?Test1
          {

          ????Test1()?//?無(wú)參構(gòu)造函數(shù)
          ????{?
          ????????cout?<"Construct?Test1"?<endl?;
          ????}

          ????Test1(const?Test1&?t1)?//?拷貝構(gòu)造函數(shù)
          ????{
          ????????cout?<"Copy?constructor?for?Test1"?<endl?;
          ????????this->a?=?t1.a?;
          ????}

          ????Test1&?operator?=?(const?Test1&?t1)?//?賦值運(yùn)算符
          ????{
          ????????cout?<"assignment?for?Test1"?<endl?;
          ????????this->a?=?t1.a?;
          ????????return?*this;
          ????}

          ????int?a?;
          };

          struct?Test2
          {

          ????Test1?test1?;
          ????Test2(Test1?&t1)
          ????{
          ????????test1?=?t1?;
          ????}
          };


          調(diào)用代碼


          Test1?t1?;
          Test2?t2(t1)?;

          輸出


          解釋一下,第一行輸出對(duì)應(yīng)調(diào)用代碼中第一行,構(gòu)造一個(gè)Test1對(duì)象。第二行輸出對(duì)應(yīng)Test2構(gòu)造函數(shù)中的代碼,用默認(rèn)的構(gòu)造函數(shù)初始化對(duì)象test1,這就是所謂的初始化階段。第三行輸出對(duì)應(yīng)Test1的賦值運(yùn)算符,對(duì)test1執(zhí)行賦值操作,這就是所謂的計(jì)算階段。

          為什么使用初始化列表

          初始化類的成員有兩種方式,一是使用初始化列表,二是在構(gòu)造函數(shù)體內(nèi)進(jìn)行賦值操作。使用初始化列表主要是基于性能問(wèn)題,對(duì)于內(nèi)置類型,如int, float等,使用初始化類表和在構(gòu)造函數(shù)體內(nèi)初始化差別不是很大,但是對(duì)于類類型來(lái)說(shuō),最好使用初始化列表,為什么呢?由上面的測(cè)試可知,使用初始化列表少了一次調(diào)用默認(rèn)構(gòu)造函數(shù)的過(guò)程,這對(duì)于數(shù)據(jù)密集型的類來(lái)說(shuō),是非常高效的。同樣看上面的例子,我們使用初始化列表來(lái)實(shí)現(xiàn)Test2的構(gòu)造函數(shù)


          struct?Test2
          {

          ????Test1?test1?;
          ????Test2(Test1?&t1):test1(t1){}

          }


          使用同樣的調(diào)用代碼,輸出結(jié)果如下。



          第一行輸出對(duì)應(yīng) 調(diào)用代碼的第一行。第二行輸出對(duì)應(yīng)Test2的初始化列表,直接調(diào)用拷貝構(gòu)造函數(shù)初始化test1,省去了調(diào)用默認(rèn)構(gòu)造函數(shù)的過(guò)程。所以一個(gè)好的原則是,能使用初始化列表的時(shí)候盡量使用初始化列表。

          哪些東西必須放在初始化列表中

          除了性能問(wèn)題之外,有些時(shí)場(chǎng)合初始化列表是不可或缺的,以下幾種情況時(shí)必須使用初始化列表


          • 常量成員,因?yàn)槌A恐荒艹跏蓟荒苜x值,所以必須放在初始化列表里面

          • 引用類型,引用必須在定義的時(shí)候初始化,并且不能重新賦值,所以也要寫在初始化列表里面

          • 沒(méi)有默認(rèn)構(gòu)造函數(shù)的類類型,因?yàn)槭褂贸跏蓟斜砜梢圆槐卣{(diào)用默認(rèn)構(gòu)造函數(shù)來(lái)初始化,而是直接調(diào)用拷貝構(gòu)造函數(shù)初始化。


          對(duì)于沒(méi)有默認(rèn)構(gòu)造函數(shù)的類,我們看一個(gè)例子。


          struct?Test1
          {

          ????Test1(int?a):i(a){}
          ????int?i?;
          };

          struct?Test2
          {

          ????Test1?test1?;
          ????Test2(Test1?&t1)
          ????{
          ????????test1?=?t1?;
          ????}
          };


          以上代碼無(wú)法通過(guò)編譯,因?yàn)門est2類中Test1 test1;需要調(diào)用默認(rèn)的構(gòu)造函數(shù),但是Test1類沒(méi)有無(wú)參的構(gòu)造函數(shù),但是由于Test1沒(méi)有默認(rèn)的構(gòu)造函數(shù),故而編譯錯(cuò)誤。正確的代碼如下,使用初始化列表代替賦值操作。


          struct?Test2
          {

          ????Test1?test1?;
          ????Test2(Test1?&t1):test1(t1){}
          }

          成員變量的初始化順序

          成員是按照他們?cè)陬愔谐霈F(xiàn)的順序進(jìn)行初始化的,而不是按照他們?cè)诔跏蓟斜沓霈F(xiàn)的順序初始化的,看代碼。


          struct?foo
          {

          ????int?i?;
          ????int?j?;
          ????foo(int?x):i(x),?j(i){};?//?ok,?先初始化i,后初始化j
          };

          再看下面的代碼


          struct?foo
          {

          ????int?i?;
          ????int?j?;
          ????foo(int?x):j(x),?i(j){}?//?i值未定義

          };


          這里i的值是未定義的,雖然j在初始化列表里面出現(xiàn)在i前面,但是i先于j定義,所以先初始化i,但i由j初始化,此時(shí)j尚未初始化,所以導(dǎo)致i的值未定義。所以,一個(gè)好的習(xí)慣是,按照成員定義的順序進(jìn)行初始化。


          版權(quán)申明:內(nèi)容來(lái)源網(wǎng)絡(luò),版權(quán)歸原創(chuàng)者所有。除非無(wú)法確認(rèn),都會(huì)標(biāo)明作者及出處,如有侵權(quán),煩請(qǐng)告知,我們會(huì)立即刪除并致歉!

          瀏覽 32
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          評(píng)論
          圖片
          表情
          推薦
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  特级毛片内射 | 亚洲AⅤ电影 | 日韩成人AV电影在线 | 黄色片网站视频看免费在线 | 色姑娘伊人 |