• <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 笨笨 閱讀(6182) 評論(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一般用于簡單類型,對于復雜的向下轉換是不確定的  回復  更多評論   

            久久无码人妻一区二区三区| 男女久久久国产一区二区三区| 大香网伊人久久综合网2020| 狠狠精品干练久久久无码中文字幕| 久久久免费观成人影院| 国内精品久久久久影院薰衣草| 青青草原1769久久免费播放| 7777精品伊人久久久大香线蕉| 国产V亚洲V天堂无码久久久| 久久性生大片免费观看性| 色综合久久久久综合体桃花网| 99久久综合国产精品二区| 久久久一本精品99久久精品88| 国产精品久久久久乳精品爆| 国产精品久久久久久福利漫画 | 久久精品无码午夜福利理论片| 夜夜亚洲天天久久| 99久久人妻无码精品系列蜜桃 | 国内精品久久久久影院薰衣草 | 热re99久久6国产精品免费| 久久免费大片| 久久影院午夜理论片无码| 99re这里只有精品热久久| 久久精品aⅴ无码中文字字幕不卡 久久精品aⅴ无码中文字字幕重口 | 久久福利青草精品资源站| 久久天堂AV综合合色蜜桃网| 亚洲欧美成人久久综合中文网| 久久久久久亚洲精品不卡| 国产一区二区精品久久凹凸| 大美女久久久久久j久久| 99久久超碰中文字幕伊人| 久久精品夜夜夜夜夜久久| 精品久久久久久无码中文字幕一区 | 一本色道久久综合亚洲精品| 中文精品99久久国产| 漂亮人妻被中出中文字幕久久| 亚洲精品国产自在久久| 久久久国产99久久国产一| 综合久久国产九一剧情麻豆| 亚洲国产精品成人久久| 久久精品嫩草影院|