• <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>
            野指針,也就是指向不可用內存區域的指針。通常對這種指針進行操作的話,將會使程序發生不可預知的錯誤。
            “野指針”不是NULL指針,是指向“垃圾”內存的指針。人們一般不會錯用NULL指針,因為用if語句很容易判斷。但是“野指針”是很危險的,if語句對它不起作用。野指針的成因主要有兩種:

            一、指針變量沒有被初始化。任何指針變量剛被創建時不會自動成為NULL指針,它的缺省值是隨機的,它會亂指一氣。所以,指針變量在創建的同時應當被初始化,要么將指針設置為NULL,要么讓它指向合法的內存。

            二、指針p被free或者delete之后,沒有置為NULL,讓人誤以為p是個合法的指針。別看free和delete的名字惡狠狠的(尤其是delete),它們只是把指針所指的內存給釋放掉,但并沒有把指針本身干掉。通常會用語句if (p != NULL)進行防錯處理。很遺憾,此時if語句起不到防錯作用,因為即便p不是NULL指針,它也不指向合法的內存塊。例:

            char *p = (char *) malloc(100);

            strcpy(p, “hello”);

            free(p); // p 所指的內存被釋放,但是p所指的地址仍然不變

            if(p != NULL) // 沒有起到防錯作用

            strcpy(p, “world”); // 出錯

            另外一個要注意的問題:不要返回指向棧內存的指針或引用,因為棧內存在函數結束時會被釋放。

            -------------------------------------------------------------------------------------------------------------------------------------------------
            野指針,也就是指向不可用內存區域的指針。通常對這種指針進行操作的話,將會使程序發生不可預知的錯誤。首先請諸位看以下一段“危險”的C++代碼:

            void function( void )
            {
            char* str = new char[100];
            delete[] str;
            // Do something
            strcpy( str, "Dangerous!!" );
            }


            之所以說其危險,是因為這是一段完全合乎語法的代碼,編譯的時候完美得一點錯誤也不會有,然而當運行到strcpy一句的時候,問題就會出現,因為在這之前,str的空間已經被delete掉了,所以strcpy當然不會成功。對于這種類似的情況,在林銳博士的書中有過介紹,稱其為“野指針”。

            那么,諸位有沒有見過安全的“野指針”呢?下面請看我的一段C++程序,靈感來自CSDN上的一次討論。在此,我只需要C++的“類”,C++的其余一概不需要,因此我沒有使用任何的C++標準庫,連輸出都是用printf完成的。

            #include <stdio.h>

            class CTestClass
            {
            public:
            CTestClass( void );
            int m_nInteger;
            void Function( void );
            };

            CTestClass::CTestClass( void )
            {
            m_nInteger = 0;
            }

            void CTestClass::Function( void )
            {
            printf( "This is a test function.\n" );
            }

            void main( void )
            {
            CTestClass* p = new CTestClass;
            delete p;
            p->Function();
            }


            OK,程序到此為止,諸位可以編譯運行一下看看結果如何。你也許會驚異地發現:沒有任何的出錯信息,屏幕上竟然乖乖地出現了這么一行字符串:

            This is a test function.

            奇怪嗎?不要急,還有更奇怪的呢,你可以把主函數中加上一句更不可理喻的:

            ((CTestClass*)NULL)->Function();


            這仍然沒有問題!!

            我這還有呢,哈哈。現在你在主函數中這么寫,倘說上一句不可理喻,那么以下可以叫做無法無天了:

            int i = 888;
            CTestClass* p2 = (CTestClass*)&i;
            p2->Function();


            你看到了什么?是的,“This is a test function.”如約而至,沒有任何的錯誤。

            你也許要問為什么,但是在我解答你之前,請你在主函數中加入如下代碼:

            printf( "%d, %d", sizeof( CTestClass ), sizeof( int ) );


            這時你就會看到真相了:輸出結果是——得到的兩個十進制數相等。對,由sizeof得到的CTestClass的大小其實就是它的成員m_nInteger的大小。亦即是說,對于CTestClass的一個實例化的對象(設為a)而言,只有a.m_nInteger是屬于a這個對象的,而a.Function()卻是屬于CTestClass這個類的。所以以上看似危險的操作其實都是可行且無誤的。

            現在你明白為什么我的“野指針”是安全的了,那么以下我所列出的,就是在什么情況下,我的“野指針”不安全:

            在成員函數Function中對成員變量m_nInteger進行操作;
            將成員函數Function聲明為虛函數(virtual)。
            以上的兩種情況,目的就是強迫野指針使用屬于自己的東西導致不安全,比如第一種情況中操作本身的m_nInteger,第二種情況中變為虛函數的Function成為了屬于對象的函數(這一點可以從sizeof看出來)。

            其實,安全的野指針在實際的程序設計中是幾乎毫無用處的。我寫這一篇文章,意圖并不是像孔乙己一樣去琢磨回字有幾種寫法,而是想通過這個小例子向諸位寫明白C++的對象實例化本質,希望大家不但要明白what和how,更要明白why。李馬二零零三年二月二十日作于自宅。


            關于成員函數CTestClass::Function的補充說明



            這個函數是一個普通的成員函數,它在編譯器的處理下,會成為類似如下的代碼:

            void Function( const CTestClass * this ) // ①
            {
            printf("This is a test function.\n");
            }


            那么p->Function();一句將被編譯器解釋為:

            Function( p );


            這就是說,普通的成員函數必須經由一個對象來調用(經由this指針激活②)。那么由上例的delete之后,p指針將會指向一個無效的地址,然而p本身是一個有效的變量,因此編譯能夠通過。并且在編譯通過之后,由于CTestClass::Function的函數體內并未對這個傳入的this指針進行任何的操作,所以在這里,“野指針”便成了一個看似安全的東西。

            然而若這樣改寫CTestClass::Function:

            void CTestClass::Function( void )
            {
            m_nInteger = 0;
            }


            那么它將會被編譯器解釋為:

            void Function( const CTestClass * this )
            {
            this->m_nInteger = 0;
            }


            你看到了,在p->Function();的時候,系統將會嘗試在傳入的這個無效地址中尋找m_nInteger成員并將其賦值為0,剩下的我不用說了——非法操作出現了。

            至于virtual虛函數,如果在類定義之中將CTestClass聲明為虛函數:

            class CTestClass
            {
            public:
            // ...
            virtual void Function( void );
            };


            那么C++在構建CTestClass類的對象模型時,將會為之分配一個虛函數表vptr(可以從sizeof看出來)。vptr是一個指針,它指向一個函數指針的數組,數組中的成員即是在CTestClass中聲明的所有虛函數。在調用虛函數的時候,必須經由這個vptr,這也就是為什么虛函數較之普通成員函數要消耗一些成本的緣故。以本例而言,p->Function();一句將被編譯器解釋為:

            (*p->vptr[1])( p ); // 調用vptr表中索引號為1的函數(即Function)③


            上面的代碼已經說明了,如果p指向一個無效的地址,那么必然會有非法操作。

            備注:

            ①關于函數的命名,我采用了原名而沒有變化。事實上編譯器為了避免函數重載造成的重名情況,會對函數的名字進行處理,使之成為獨一無二的名稱。
            ②將成員函數聲明為static,可以使成員函數不經由this指針便可調用。
            ③vptr表中,索引號0為類的type_info。


            posted on 2007-11-09 22:21 清源游民 閱讀(1907) 評論(1)  編輯 收藏 引用 所屬分類: C++

            FeedBack:
            # re: (網摘好文)關于野指針
            2007-11-14 23:01 | <a href=http://minidx.com>minidxer</a>
            其實野指針產生于錯誤的/不嚴謹的邏輯,  回復  更多評論
              
            <2007年3月>
            25262728123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            留言簿(35)

            隨筆分類(78)

            隨筆檔案(74)

            文章檔案(5)

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            亚洲va久久久噜噜噜久久狠狠 | 国产ww久久久久久久久久| 国产成年无码久久久免费| 久久国语露脸国产精品电影| 少妇人妻88久久中文字幕| 久久久噜噜噜久久中文福利| 国产综合久久久久久鬼色| 久久久久久无码Av成人影院| 久久精品国产精品国产精品污| 久久国产精品国语对白| 久久久一本精品99久久精品88| 99re久久精品国产首页2020| 日日狠狠久久偷偷色综合免费| 久久这里只有精品18| 久久久精品国产Sm最大网站| avtt天堂网久久精品| 狠狠色丁香久久婷婷综合图片| 欧美久久综合性欧美| 2021国内久久精品| 久久精品一区二区| 久久久久国产精品熟女影院| 久久久无码精品亚洲日韩蜜臀浪潮| 青青青青久久精品国产| 久久国产亚洲精品无码| 久久经典免费视频| 久久乐国产精品亚洲综合| 欧美丰满熟妇BBB久久久| 狠狠色丁香久久婷婷综合_中| 久久国产一区二区| 久久青青草原精品国产| 亚洲中文字幕无码久久精品1| 一本久久综合亚洲鲁鲁五月天亚洲欧美一区二区| 国产成人久久AV免费| 浪潮AV色综合久久天堂| 久久99久久99精品免视看动漫| 亚洲欧美日韩久久精品| 久久婷婷五月综合色99啪ak| 日韩va亚洲va欧美va久久| 久久亚洲色一区二区三区| 精品久久久久久久久久中文字幕 | 久久精品成人欧美大片|