• <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>

            小星星的天空

            O(∩_∩)O 小月亮的fans ^_^

              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理 ::
              16 隨筆 :: 0 文章 :: 61 評論 :: 0 Trackbacks
            通常在C的編程中,我們經常使用memset函數將一塊連續的內存區域清零或設置為其它指定的值,最近在移植一段java代碼到C++的時候,不當使用memset函數花費了我幾個小時的調試時間。對于虛函數的底層機制很多資料都有較詳細闡述,但對我個人而言,這次的調試讓我感觸頗深。

            先來看一段代碼,在繼承的類Advance之中,有很多屬性字段,我希望將其清成0或NULL,于是在構造函數中我通過memset將當前類的所有屬性置0。

            class Base{

            public:

            virtual void kickoff() = 0;

            };
            class Advance:public Base{

            public:

            Advance(){

            memset(this, 0, sizeof(Advance));

            }

            void kickoff(){

            count++;

            //... do something else;

            }

            private:

            int attr1, attr2;

            char* label;

            int count;

            //... other attributes, they should be initiated to 0 or NULL at beginning.

            };

            int _tmain(int argc, _TCHAR* argv[])

            {
            Base* ptr = new Advance();
            ptr->kickoff();
            return 0;
            }

            這樣看似能正常運行,但運行程序時,你會發現類似于下面的錯誤:

            TestVirtual.exe 中的 0x00415390 處未處理的異常: 0xC0000005: 讀取位置 0x00000000 時發生訪問沖突

            同時斷點停留在ptr->kickoff()處,從錯誤提示我們可以得知無法調用kickoff方法,這個方法的指針沒有被正確初始化,但為什么呢?

            指出問題之前,先看看這段文獻上的關于虛函數機制的說明:

            函數賴以生存的底層機制:vptr + vtable。虛函數的運行時實現采用了VPTR/VTBL的形式,這項技術的基礎:
            ①編譯器在后臺為每個包含虛函數的類產生一個靜態函數指針數組(虛函數表),在這個類或者它的基類中定義的每一個虛函數都有一個相應的函數指針。
            ②每個包含虛函數的類的每一個實例包含一個不可見的數據成員vptr(虛函數指針),這個指針被構造函數自動初始化,指向類的vtbl(虛函數表)
            ③當客戶調用虛函數的時候,編譯器產生代碼反指向到vptr,索引到vtbl中,然后在指定的位置上找到函數指針,并發出調用。

            這里的問題,就出在

            memset(this, 0, sizeof(Advance));

            上面,虛函數指針應該在進入構造函數賦值體之前自動初始化的,而memset卻又將已經初始化好的指針清0了,這就是為什么會產生上面的訪問零址的錯誤。將上面的memset語句去除程序就可以正常運行了。

            所以,從上面的問題中,我們可以看出在構造函數體內調用memset將整個對象清0是很有風險的,當沒有虛函數的時候上面程序可以正常運行(可以試著將Base類的純虛函數聲明改成非虛函數再運行程序)。初始化類的屬性對象時,比較穩妥的辦法還是手動逐個進行初使化
            posted on 2009-10-20 21:11 Little Star 閱讀(2809) 評論(7)  編輯 收藏 引用 所屬分類: 找工作

            評論

            # re: 【轉】不當使用memset函數帶來的麻煩問題 2009-10-20 22:32 OwnWaterloo
            沒有虛函數也不可以亂來。
            空指針并不一定是二進制全0。

            1.
            char* label = 0;

            2.
            char* label;
            memset(&label,0,sizeof(label) );

            有平臺上兩者功能不同。

              回復  更多評論
              

            # re: 【轉】不當使用memset函數帶來的麻煩問題 2009-10-21 10:47 Little Star
            謝謝!@OwnWaterloo
              回復  更多評論
              

            # re: 【轉】不當使用memset函數帶來的麻煩問題 2009-10-21 21:11 周龍亭
            在C++中不能用memset來初始化一個類。
            你上面的的代碼,其實是用寫C代碼的習慣來寫C++代碼,
            推薦LZ全新的學習C++。  回復  更多評論
              

            # re: 【轉】不當使用memset函數帶來的麻煩問題 2009-10-21 23:02 Little Star
            @周龍亭

            這個是我轉的,不代表我的意見。
            問題是如果說一個類有上千個屬性的話,要咋清零呢?
              回復  更多評論
              

            # re: 【轉】不當使用memset函數帶來的麻煩問題 2009-10-21 23:39 OwnWaterloo
            @Little Star
            class C {
            /* data declaration */
            public:
            C() { memset(this,0,sizeof(*this); }
            };

            改為:
            class C {
            struct data {
            /* data declaration */
            } data_;
            public:
            C() { memset(&data_,0,sizeof(data_); }
            };


            還是需要注意memset( ... 0 ... );
            不能保證: 指針是nullptr,浮點數是0.0, 0.0f, 0.0lf。
            能保證:整數是0, 字符是null字符,即'\0'。

              回復  更多評論
              

            # re: 【轉】不當使用memset函數帶來的麻煩問題[未登錄] 2009-10-22 01:52 Little star
            @OwnWaterloo
            那,有誰能講講為什么不能保證:指針是nullptr,浮點數是0.0, 0.0f, 0.0lf呢?  回復  更多評論
              

            # re: 【轉】不當使用memset函數帶來的麻煩問題 2009-10-22 02:40 OwnWaterloo
            @Little star
            標準就這么規定的。

            《C 語言常見問題集》 5.14中介紹了一些古怪的空指針。
            “至少PL/I, Prime 50 系列用段07777, 偏移0 作為空指針。
            ……
            CDC Cyber 180 系列使用包含環(ring), 段和位移的48 位指針。多數用戶
            (在環11 上) 使用的空指針為0xB00000000000。
            在舊的1 次補碼的CDC 機器上
            用全1 表示各種數據, 包括非法指針, 是十分常見的事情。
            Symbolics Lisp 機器是一種標簽結構, 它甚至沒有傳統的數字指針; 它使用
            <NIL, 0> 對(通常是不存在的<對象, 偏移> 句柄) 作為C 空指針。


            浮點如果是采用IEEE754, 0.0恰好是二進制全0。
            但標準沒有保證浮點數一定采用IEEE754。

              回復  更多評論
              

            久久久久综合中文字幕| 久久SE精品一区二区| 国产精品无码久久四虎| 欧美与黑人午夜性猛交久久久| 亚洲午夜无码AV毛片久久| 欧美亚洲色综久久精品国产| 久久精品免费一区二区三区| 国色天香久久久久久久小说| 91精品观看91久久久久久| 国产亚洲精久久久久久无码77777| 国内精品久久久久久野外| 中文字幕亚洲综合久久菠萝蜜| 国产精品女同久久久久电影院 | 少妇久久久久久被弄高潮| 韩国三级大全久久网站| 狠狠色噜噜色狠狠狠综合久久| 国产午夜福利精品久久| 久久久久亚洲av无码专区喷水| 午夜精品久久久久久| 99久久国产免费福利| 欧美va久久久噜噜噜久久| 久久免费视频1| 久久久久久无码国产精品中文字幕| 久久精品九九亚洲精品| 一本一本久久A久久综合精品 | 亚洲精品97久久中文字幕无码| 2021精品国产综合久久| 久久综合香蕉国产蜜臀AV| 亚洲伊人久久综合中文成人网| 久久精品国产精品亚洲| 国产精品热久久毛片| 日韩一区二区久久久久久| 久久精品人人槡人妻人人玩AV| 午夜精品久久久久久久久| 久久精品国产亚洲av麻豆蜜芽| 欧美久久久久久| 久久这里有精品| 7777精品久久久大香线蕉| 亚洲色欲久久久综合网| 久久精品欧美日韩精品| 久久精品国产半推半就|