• <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>
            Dict.CN 在線詞典, 英語學習, 在線翻譯

            學海苦作舟,書山勤為徑

            留下點回憶

            常用鏈接

            統計

            積分與排名

            Denoise

            English study

            Web技術

            數據壓縮

            一些連接

            最新評論

            dynamic_cast介紹

            dynamic_cast<type-id> (expression)

            這個表達式將 expression 轉換為一個 type-id 類型的對象。 Type-id 必須是一個指針、指向一個已經定義類的類型或一個指向 VOID 的指針。 Expression 的類型必須是一個指針,如果 type-id 是一個指針;當 type-id 是一個引用的時候必須是一個左值。

            如果 type-id 是一個到 expression 類的直接或間接的模棱兩可的指針,結果是一個到 type-id 類型的子對象:

            class B { ... };

            class C : public B { ... };

            class D : public C { ... };

            ?

            void f(D* pd)

            {

            ?? C* pc = dynamic_cast<C*>(pd);?? // ok: C 是一個直接的基類

            ???????????????????????????????? // pc 指向 pd C 子對象

            ?

            ?? B* pb = dynamic_cast<B*>(pd);?? // ok: B 是一個間接的基類

            ??????????????????????????? ?????// pb 指向 pd B 子對象

            ?? ...

            }

            這個類型轉換叫做向上轉型,因為它將一個指針在其繼承層次向上轉型,即從一個繼承類到其基類。向上轉型是隱式轉換。

            如果 type-id 是一個 void* ,運行時檢查將決定表達式的實際類型。結果是一個到 expression 指向的完整對象。例如:

            class A { ... };

            ?

            class B { ... };

            ?

            void f()

            {

            ?? A* pa = new A;

            ?? B* pb = new B;

            ?? void* pv = dynamic_cast<void*>(pa);

            ?? // pv 指向一個 A 類型的對象

            ?? ...

            ?? pv = dynamic_cast<void*>(pb);

            ?? // pv 指向一個 B 類型的對象

            }

            如果 type-id 不是 void* ,運行時檢查指向 expression 的對象能否轉換為指向 type-id 類型的對象。

            如果 expression 類型是 type-id 的基類,運行時檢查是否 expression 實際是一個指向 type-id 類型的完整對象,如果是,結果返回指向 type-id 類型的完整對象,否則返回 NULL 。例如:

            class B { ... };

            class D : public B { ... };

            void f()

            {

            ?? B* pb = new D;???????????????????? // unclear but ok

            ?? B* pb2 = new B;

            ?? D* pd = dynamic_cast<D*>(pb);????? // ok: pb 實際指向 D

            ?? ...

            ?? D* pd2 = dynamic_cast<D*>(pb2);?? // pb2 實際指向 B 而不是 D

            ????????? ??????????????????????????// 轉換失敗, pd2 NULL

            ?? ...

            }

            向下類型轉換之所以這么說是因為其從類繼承層次的父類向子類轉換。

            在多重繼承的情況,可能導致二義性。看一下下面的類繼承層次:

            H1.bmp


            指向類型D的指針轉換為BC都正常,但如果從D轉換到A將會怎么樣來?這個結果導致轉換的二義性錯誤;為了結果這個問題,你可以指向兩次明確的轉型,例如:

            void f()

            {

            ?? D* pd = new D;

            ?? A* pa = dynamic_cast<A*>(pd);??? ??// 錯誤:二義性

            ?? B* pb = dynamic_cast<B*>(pd);????? // 首先轉換到 B

            ?? A* pa2 = dynamic_cast<A*>(pb);?? // ok: 明確的

            }

            在使用虛基類的時候就導致更復雜的模糊;看下面的類層次圖:

            H2.bmp


            在這個繼承層次中,A是虛基類。假定一個類E的實例并且一個指向A子對象的指針,一次到Bdynamic_cast會由于不明確性導致失敗,你必須首先轉換到適當的層次,然后再向上轉換到確定的層次,一直按照這種方式直到到達正確的B對象。

            看下面的類層次圖:

            H3.bmp


            假定一個類型E的對象和一個指向D子對象的指針,從D子對象導航到左上A子對象,必須執行三個轉換。從DEdynamic_cast的轉換,然后一個從EB的轉換(可以是dynamic_cast或者隱式轉換),最終是從BA的轉換,例如:

            void f(D* pd)

            {

            ?? E* pe = dynamic_cast<E*>(pd);// 這里的 D 實際上是 E 類型的對象

            ?? B* pb = pe;????? // upcast, implicit conversion

            ?? A* pa = pb;????? // upcast, implicit conversion

            }

            dynamic_cast 操作能執行交叉轉換,使用上面相同的類層次,從 B 子對象到 D 子對象轉換是可能的,只要完整的對象是 E

            由于交叉轉換,從 D 指針到左上角 A 子對象的指針是可行的;首先從 D B 的交叉轉換,然后隱式從 B A 的轉換。例如:

            void f(D* pd)

            {

            ?? B* pb = dynamic_cast<B*>(pd);????? // cross cast

            ?? A* pa = pb;?????????????? ???// upcast, implicit conversion

            }

            一個 NULL 指針值通過 dynamic_cast 轉換到一個 NULL 指針。

            當使用 dynamic_cast < type-id > ( expression ) 時, 如果 expression 不能安全的轉換到 type-id ,運行時檢查導致轉型失敗,例如:

            class A { ... };

            ?

            class B { ... };

            ?

            void f()

            {

            ?? A* pa = new A;

            ?? B* pb = dynamic_cast<B*>(pa);????? // fails, not safe;

            ??????????????????????????????????? // B not derived from A

            ?? ...

            }

            轉換失敗的指針類型是 NULL 指針。失敗的引用類型轉換拋出 bad_cast_exception 異常;如果 expression 沒有指向或引用一個有效的對象將拋出 __non_rtti_object 異常。

            posted on 2006-03-28 22:25 笨笨 閱讀(6186) 評論(2)  編輯 收藏 引用

            評論

            # re: dynamic_cast介紹 2006-03-30 14:46 小石

            是否可以這樣理解
            dynamic_cast主要設計目的是類對象指針和類對象左值的向下裝換?
            static_cast則主要用于簡單數據類型之間的轉換?  回復  更多評論   

            # re: dynamic_cast介紹 2006-03-30 20:10 笨笨

            dynamic_cast保證安全的轉換,因為如果不成功返回的是NULL,而static_cast一般用于簡單類型,對于復雜的向下轉換是不確定的  回復  更多評論   

            久久精品国产精品亚洲精品| 国内精品久久久久久99蜜桃| 国产精品久久久久aaaa| 中文字幕无码免费久久| 亚洲а∨天堂久久精品| 久久人人爽人人爽人人片AV麻豆| AAA级久久久精品无码区| 久久99精品久久久久久动态图| 日韩乱码人妻无码中文字幕久久| 五月丁香综合激情六月久久| 久久精品一区二区三区AV| 精品国产乱码久久久久久呢| 欧美va久久久噜噜噜久久| 久久男人Av资源网站无码软件| 久久精品无码专区免费东京热| 久久97精品久久久久久久不卡| 91久久成人免费| 久久亚洲2019中文字幕| 久久无码专区国产精品发布 | 99久久国产宗和精品1上映| 久久婷婷人人澡人人爽人人爱| 久久久久久伊人高潮影院| 狼狼综合久久久久综合网| 久久国产亚洲精品麻豆| 久久久久国产一级毛片高清板| 久久久久久免费视频| 999久久久免费精品国产| 久久久久人妻一区精品| 无码日韩人妻精品久久蜜桃| 伊人久久综合热线大杳蕉下载| 波多野结衣久久精品| 精品久久久久久国产| 久久亚洲国产最新网站| 狠狠色丁香婷婷综合久久来| 欧美日韩精品久久久久| av午夜福利一片免费看久久| 久久精品国产精品亚洲下载| 精品无码久久久久久尤物| 99久久国产亚洲综合精品| 伊人久久大香线焦综合四虎| 一本一本久久a久久综合精品蜜桃|