• <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>
            酸菜豬蹄的程序人生
            木下編程屯屯燙燙
            C++異常捕獲是在運行時進行的,但是拋出的對象卻是在編譯時確定的,編譯時會對拋出的對象上溯查找到無二義性的基類,然后拋出這個對象的引用。
             
            例如:
            #include <iostream>
            using namespace std;
            class CBase
            {
            public:
             virtual ~CBase(){};
            };
             
            class CDerived:public CBase
            {
            };
             
            void exceMaker()
            {
                throw CDerived();
            }
             
            void exceCatcher()
            {
                try
                {
                    exceMaker();
                }
                catch(CBase&)
                {
                    cout << "caught a CBase" << endl;
                }
                catch(...)
                {
                    cout << "caught else" << endl;
                }
            }
             
            int main(int argc, char** argv)
            {
                exceCatcher();
                cin.get();
                return 0;
            }
             
            運行后將打印出:
             
            caught a CBase.
             
            編譯器進行了類型上溯轉換,拋出的CBase的引用,如果同時捕獲CDerived&,也即在exce中加入如下代碼:
            catch(CDerived&)
            {
                cout << "caught a CDerived" << endl;
            }
            編譯時將會給出warning,說異常已經被 catch(CBase&)捕獲,證明在編譯時進行了轉換。
             

            而如果修改CDerived 為私有繼承CBase,整體代碼如下:
               
            #include <iostream>
            using namespace std;
            class CBase
            {
            public:
             virtual ~CBase(){};
            };
             
            class CDerived:private CBase
            {
            };
             
            void exceMaker()
            {
                throw CDerived();
            }
             
            void exceCatcher()
            {
                try
                {
                    exceMaker();
                }
                catch(CBase&)
                {
                    cout << "caught a CBase" << endl;
                }
                catch(...)
                {
                    cout << "caught else" << endl;
                }
            }
             
            int main(int argc, char** argv)
            {
                exceCatcher();
                cin.get();
                return 0;
            }   
             
            將打印出"caught else";
            因為私有繼承后,exceMaker函數不能對私有繼承的基類進行上溯(private權限限制),所以拋出的異常為CDerived&,不再是CBase&.
             
            而如果這樣:
             
            #include <iostream>
            using namespace std;
            class CBase
            {
            public:
             virtual ~CBase(){};
            };
             
            class CDerived:private CBase
            {
                friend void exceMaker();
            };
             
            void exceMaker()
            {
                throw CDerived();
            }
             
            void exceCatcher()
            {
                try
                {
                    exceMaker();
                }
                catch(CBase&)
                {
                    cout << "caught a CBase" << endl;
                }
                catch(...)
                {
                    cout << "caught else" << endl;
                }
            }
             
            int main(int argc, char** argv)
            {
                exceCatcher();
                cin.get();
                return 0;
            }
             
            在VC6中將打印出"caught CBase",因為exceMaker是CDerived的友元函數,可以訪問它的私有成員,故可以上溯到CBase&,但后續的編譯器版本已經更正為caught else. 因為不是ISA關系。
             
             
            posted on 2006-01-06 17:51 cooelaf 閱讀(3575) 評論(4)  編輯 收藏 引用 所屬分類: Pure C/C++
            Comments
            • # re: C++異常捕獲機制
              小明
              Posted @ 2006-01-09 12:26
              你說:
              因為exceMaker是CDerived的友元函數,可以訪問它的私有成員,故可以上溯到CBase&

              這種說法太牽強,我不能理解。

              我認為這是vc6/vs2005的C++ exception實現的一個Bug。
              你的最后一個例子 我在C++ BuilderX下面測試的結果是:
              caught else
                回復  更多評論   
            • # re: C++異常捕獲機制
              cooelaf
              Posted @ 2006-01-10 08:32
              to 小明:
              我最開始的時候也是這樣想,因為最后一種繼承方式是私有繼承,不是isa關系,所以不應該上溯到CBase,如果如你所說,可能是vs系列的bug,我會查找一下c++標準對這方面的說法。謝謝。  回復  更多評論   
            • # re: C++異常捕獲機制
              yskin
              Posted @ 2006-02-22 11:04
              最后一個:
              gcc
              caught else  回復  更多評論   
            • # re: C++異常捕獲機制
              cooelaf
              Posted @ 2006-02-22 16:55
              看來確實是這樣。  回復  更多評論   
             
            一本色道久久88综合日韩精品| 久久综合综合久久狠狠狠97色88| 久久久久这里只有精品| 国产成人综合久久精品红| 久久99精品国产麻豆宅宅| 色综合久久88色综合天天 | 久久亚洲AV成人无码软件| 国产精品青草久久久久福利99 | 久久久久黑人强伦姧人妻| 国产成人精品综合久久久| 国产精品永久久久久久久久久 | 久久国产一片免费观看| 精品午夜久久福利大片| 国内精品人妻无码久久久影院导航 | 精品久久久久久国产三级| 久久亚洲中文字幕精品有坂深雪| 国产伊人久久| 高清免费久久午夜精品| 久久久亚洲欧洲日产国码二区| 久久久久久久久66精品片| 久久久久亚洲AV无码专区桃色 | 久久精品人人做人人爽97| 亚洲人成电影网站久久| 亚洲国产成人久久精品99| 国内精品久久久久久久亚洲| 久久精品视频网| 久久久久亚洲AV成人片| 青草国产精品久久久久久| 欧美午夜精品久久久久免费视| 久久中文字幕视频、最近更新| 99精品伊人久久久大香线蕉| 久久综合中文字幕| 久久精品国产免费一区| 久久se精品一区精品二区国产| 99久久精品无码一区二区毛片| 成人免费网站久久久| 久久久久综合国产欧美一区二区| 久久久久久久综合综合狠狠| 中文字幕亚洲综合久久菠萝蜜| 伊人久久大香线蕉av不变影院| 无遮挡粉嫩小泬久久久久久久|