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

            精品综合久久久久久88小说| 久久人人爽人人爽人人AV东京热 | 久久五月精品中文字幕| 久久福利青草精品资源站免费| 国产免费福利体检区久久| 亚洲七七久久精品中文国产| 欧美午夜精品久久久久免费视| 久久免费精品视频| 久久天天躁狠狠躁夜夜2020老熟妇 | 久久se精品一区精品二区| 久久人搡人人玩人妻精品首页| 蜜臀av性久久久久蜜臀aⅴ| 国产成人综合久久久久久| 91麻豆国产精品91久久久| 91久久成人免费| 久久久国产精华液| 久久有码中文字幕| 好属妞这里只有精品久久| 久久精品国产99国产精品亚洲| 国内精品免费久久影院| 久久久久亚洲精品无码蜜桃| 麻豆国内精品久久久久久| 91精品国产乱码久久久久久| 久久久久亚洲AV片无码下载蜜桃| 国产精品久久久久乳精品爆| 国产成人精品白浆久久69| 久久天天躁狠狠躁夜夜躁2O2O| 性做久久久久久久久老女人| 久久精品18| 久久久WWW成人| 伊人久久精品线影院| 久久精品视频网| 国产一区二区三区久久| 97久久国产亚洲精品超碰热| 日韩精品久久久久久免费| 亚洲AV无码久久精品成人| 97精品国产97久久久久久免费| 久久亚洲AV无码精品色午夜麻豆| 久久国产精品无| 亚洲精品无码久久千人斩| 亚洲精品美女久久777777|