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

            洛譯小筑

            別來(lái)無(wú)恙,我的老友…
            隨筆 - 45, 文章 - 0, 評(píng)論 - 172, 引用 - 0
            數(shù)據(jù)加載中……

            [ECPP讀書(shū)筆記 條目36] 避免對(duì)派生的非虛函數(shù)進(jìn)行重定義

            現(xiàn)在考慮以下的層次結(jié)構(gòu):B是一個(gè)基類(lèi),D是由B的公共繼承類(lèi),B類(lèi)中定義了一個(gè)公有成員函數(shù)mf,由于這里mf的參數(shù)和返回值不是討論的重點(diǎn),因此假設(shè)mf是無(wú)參數(shù)無(wú)返回值的函數(shù)。即:

            class B {

            public:

              void mf();

              ...

            };

            class D: public B { ... };

            即使不知道BDmf的任何信息,我們聲明一個(gè)D的對(duì)象x

            D x;                               // x D類(lèi)型的對(duì)象

             

            B *pB = &x;                        // 指向x的指針

            pB->mf();                          // 通過(guò)指針調(diào)用mf函數(shù)

             

            D *pD = &x;                        // 指向x的指針

            pD->mf();                          // 通過(guò)指針調(diào)用mf函數(shù)

            在這里,如果告訴你pD->mf()pB->mf()將擁有不同的行為,你很可能會(huì)感到意外。因?yàn)閮煞N情況都調(diào)用了x對(duì)象的成員函數(shù)mf,兩次調(diào)用使用了同一函數(shù)和同一對(duì)象,mf()理所應(yīng)當(dāng)具有一致的行為。難道不是嗎?

            你說(shuō)得沒(méi)錯(cuò),的確“理所應(yīng)當(dāng)”。但這一點(diǎn)無(wú)法得到保證。在特殊情況下,如果mf是非虛函數(shù)并且D類(lèi)中對(duì)mf進(jìn)行了重定義,那么問(wèn)題就出現(xiàn)了:

            class D: public B {

            public:

              void mf();                       // 隱藏了B::mf; 參見(jiàn)條目33

              ...

            };

             

            pB->mf();                          // 調(diào)用B::mf

             

            pD->mf();                          // 調(diào)用D::mf

            此類(lèi)“雙面行為”的出現(xiàn),究其原因,是由于諸如B::mfD::mf這樣的非虛函數(shù)是靜態(tài)綁定的(參見(jiàn)條目37)。這也就意味著:由于我們將pB聲明為指向B的指針,那么通過(guò)pB所調(diào)用的所有非虛函數(shù)都將調(diào)用B類(lèi)中的版本,即使pB指向一個(gè)B的派生類(lèi)的對(duì)象也是如此,正如上文示例所示。

            另一方面,由于虛函數(shù)是動(dòng)態(tài)綁定的(再次參見(jiàn)條目37),因此它們不會(huì)被這個(gè)問(wèn)題困擾。如果mf是虛函數(shù),那么無(wú)論通過(guò)pB還是pD來(lái)調(diào)用mf都會(huì)是對(duì)D::mf的調(diào)用,這是因?yàn)?span style="font-family:"Courier New";">pB和pD實(shí)際上指向同一對(duì)象,這個(gè)對(duì)象是D類(lèi)型的。

            如果你正在編寫(xiě)D類(lèi),并且你對(duì)由B類(lèi)繼承而來(lái)的mf函數(shù)進(jìn)行了重定義,那么D類(lèi)將會(huì)表現(xiàn)出不穩(wěn)定的行為。在特定情況下,任意給定的D對(duì)象在調(diào)用mf函數(shù)時(shí)可能表現(xiàn)出BD兩種不同的行為,決定因素將是指向mf的指針的類(lèi)型,與對(duì)象本身沒(méi)有任何關(guān)系。引用同樣會(huì)遭遇這種令人困惑的行為。

            但是,上述內(nèi)容僅僅是實(shí)用層面的分析,我知道,你真正需要的是對(duì)“避免對(duì)派生的非虛函數(shù)進(jìn)行重定義”這一命題的理論推導(dǎo)。我很樂(lè)意效勞。

            條目32解釋了公共繼承意味著“A是一個(gè)B”,條目34描述了為什么在類(lèi)中聲明一個(gè)非虛函數(shù)是對(duì)類(lèi)本身設(shè)置的“個(gè)性化壁壘”。將上述理論應(yīng)用到類(lèi)BD和非虛你函數(shù)B::mf上,我們可以得到:

            對(duì)B生效的所有東西對(duì)D也生效,這是因?yàn)槊恳粋€(gè)D對(duì)象都是一個(gè)B對(duì)象。

            繼承自B的類(lèi)必須同時(shí)繼承mf的接口實(shí)現(xiàn),這是因?yàn)?span style="font-family:"Courier New"">mf是B類(lèi)中的非虛函數(shù)。

            現(xiàn)在,如果在D中重定義了mf,那么你的設(shè)計(jì)方案中就出現(xiàn)了一個(gè)矛盾。如果D確實(shí)需要與B不同的mf實(shí)現(xiàn)方案,與此同時(shí),如果對(duì)于所有的B對(duì)象(無(wú)論多么個(gè)性化的)確實(shí)必須使用B實(shí)現(xiàn)版本的mf,于是我們可以很簡(jiǎn)單地推斷出:并不是每個(gè)D都是一個(gè)B。這種情況下,D并非公共繼承自B。另一方面,如果D確實(shí)必須是B的公共繼承類(lèi),與此同時(shí),如果D確實(shí)需要與B不同的mf實(shí)現(xiàn)版本,那么mf對(duì)B的“個(gè)性化壁壘”作用就不復(fù)存在了。這種情況下,mf應(yīng)該是虛函數(shù)。最后,如果每個(gè)D確實(shí)是一個(gè)B,與此同時(shí),如果mf確實(shí)對(duì)B起到了“個(gè)性化壁壘”的作用,那么D中并不會(huì)真正需要重定義mf,它也不應(yīng)該做出這樣的嘗試。

            無(wú)論從哪個(gè)角度講,我們都必須無(wú)條件地禁止對(duì)派生的非虛函數(shù)進(jìn)行重定義。

            如果閱讀本文給你一種似曾相識(shí)的感覺(jué),那么你一定是對(duì)閱讀過(guò)的條目7還有印象,在那里,我們解釋了為什么多態(tài)基類(lèi)的析構(gòu)函數(shù)必須為虛函數(shù)。如果你違背了條目7的思想(也就是說(shuō),你在多態(tài)基類(lèi)中聲明了一個(gè)非虛構(gòu)函數(shù)),那么你也就同時(shí)違背了本條的思想。這是因?yàn)樵谂缮?lèi)中繼承到的非虛函數(shù)(基類(lèi)的析構(gòu)函數(shù))一定會(huì)被重定義。即使派生類(lèi)中不聲明任何析構(gòu)函數(shù)也是如此,這是因?yàn)椋瑢?duì)于一些特定的函數(shù),即使你不自己聲明它們,編譯器也會(huì)自動(dòng)為你生成(參見(jiàn)條目5)。從本質(zhì)上講,條目7只不過(guò)是本條的一個(gè)特殊情況,只是因?yàn)樗种匾覀儾虐阉鼏瘟谐鲆粋€(gè)條目。

            時(shí)刻牢記

            避免在派生類(lèi)中重定義非虛函數(shù)。

            posted on 2011-12-30 22:53 ★ROY★ 閱讀(2274) 評(píng)論(1)  編輯 收藏 引用 所屬分類(lèi): Effective C++

            評(píng)論

            # re: 【讀書(shū)筆記】[Effective C++第3版][第36條]避免對(duì)派生的非虛函數(shù)進(jìn)行重定義  回復(fù)  更多評(píng)論   

            如果在D中重定義了mf(),那么派生類(lèi)中的mf()會(huì)屏蔽掉基類(lèi)中mf(),想調(diào)用基類(lèi)的mf(),只能通過(guò)B::mf();這樣調(diào)用了
            2012-01-12 16:33 | mc_ran
            久久99精品久久久久久噜噜 | 91久久精品91久久性色| 久久亚洲春色中文字幕久久久| 性欧美大战久久久久久久久| 女人香蕉久久**毛片精品| 久久午夜夜伦鲁鲁片免费无码影视| 久久精品亚洲一区二区三区浴池 | 精品熟女少妇AV免费久久| 久久电影网2021| 久久精品国产2020| 精品国产综合区久久久久久 | 久久人人爽人人爽人人片AV高清| 久久精品亚洲精品国产色婷| 色婷婷综合久久久久中文字幕| 久久久久亚洲av无码专区喷水| 日韩久久无码免费毛片软件| 天天综合久久久网| 久久精品中文字幕无码绿巨人| 久久天天躁狠狠躁夜夜avapp | 欧美一区二区三区久久综合| 久久精品国产欧美日韩99热| 品成人欧美大片久久国产欧美...| 精品久久久久中文字幕日本| 欧美精品国产综合久久| 亚洲人AV永久一区二区三区久久| 国产精品青草久久久久福利99 | 国产精品综合久久第一页| 99久久er这里只有精品18| 欧洲人妻丰满av无码久久不卡| 久久久久久精品免费看SSS| 麻豆久久| 伊人色综合久久天天人守人婷 | 婷婷伊人久久大香线蕉AV| 热综合一本伊人久久精品| 2021国内精品久久久久久影院| 尹人香蕉久久99天天拍| 中文字幕精品久久久久人妻| 久久久无码精品亚洲日韩京东传媒 | 狠狠色丁香久久婷婷综合| 亚洲午夜久久久久久久久久| 久久ZYZ资源站无码中文动漫|