• <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>
            隨筆 - 55  文章 - 15  trackbacks - 0
            <2012年2月>
            2930311234
            567891011
            12131415161718
            19202122232425
            26272829123
            45678910

            常用鏈接

            留言簿

            隨筆分類

            隨筆檔案

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            1. 取代#define 進行值替代
            2. 修飾指針
            3. 修飾參數和返回值
            4. 修飾類和類成員變量及類成員函數


            一、值替代
                  #define只在預處理器進行值替代,并且沒有類型標識,出現錯誤很難排查。const 提供類型信息,在編譯期可以進行類型檢查。編譯器在編譯時可以通過必要的運算把一個復雜的常量表達式簡化,稱為常量折疊。
                   我理解的常量折疊是:編譯器會把這個常量放到一個符號表中,比如說const int i = 1;符號表中有一個符號i,同時對應了一個數值1,此時是不分配內存空間的。如果你強行分配內存空間的話也可以。
            const int i = 1;
            int* ip = const_cast<int*>(&i);
            *ip = 2;
            cout <<"i = "<<i ;
            cout <<"*ip = "<< *ip;
            cout <<"*(&i) = "<<*(&i) ;
            // i = 1; 
            //*ip = 2;
            // 
            *(&i) = 1;
            不論你怎么通過內存修改i,看起來i有兩個值,1和2,這貌似就是常量折疊。
            內部鏈接問題
              C++中的const是內部鏈接,即const僅在被定義過的文件中才是可見的,鏈接時不能被其他編譯單元看到。在定義時,必須賦一個初值,不然必須用extern顯式地聲明。
            通常,編譯器是不為const創建存儲空間的,相反把這個定義保存在它的符號表中。但是extern強制執行了存儲空間的分配,因為extern表示使用外部鏈接,也就是有幾個不同的編譯單元可以引用它,所以必須有存儲空間。(這句話我的理解是:每個編譯單元都有自己的符號表,其他編譯單元是不能看到其他編譯單元符號表中的東西的。其他編譯單元如果要引用一個外部變量,該變量必須是在內存中,可以讀取的。所以一定要分配空間。外部鏈接一般是找到相應名字的內存地址,比如鏈接一個外部函數,就是找到這個函數的內存地址)
              通常沒有extern的時候,不會分配存儲空間。但是有些結構比較復雜的const,編譯器還是會分配空間。因為.h文件中一般放置的都是不分配空間的代碼,所以cosnt變量默認為內部鏈接。不然鏈接程序在多個目標文件中看到相同的定義就會”抱怨“。

            用常量表達式定義數組大小
              定義數組的大小的時候,必須是常量表達式,不能為變量,至于為什么。不太清楚,只是編譯器在編譯的時候給出錯誤:期望一個常量表達式,不能分配一個長度是0的數組,變量是長度未知的。(我的理解是,因為變量是變的,如果靜態地聲明了一個數組a[b]因為b確定,定義的時候b是1,在之后又被修改為2,那數組a長度到底應該是1呢還是2呢?)
              常量表達式:
              const int i = 100;// constant
                  const int j = 100 + i;// constant express
                  long addr = (long)&j;// forces storage
                 char buf[j + 10];//right
                 const可以應用于集合,但是編譯器不會把一個集合保存到符號表中,所以必須分配內存。這種情況下,const意味著‘不能改變的一塊內存’,編譯器不需要知道里面存儲的內容。如果這樣理解的話,那么編譯器在看到int b = 1;這句話的時候,也會想”這是一個int型的變量,我給他分配2個字節就行了,里面是什么,或許要到運行的時候再說了“,所以a[b]的長度就是未知長度了。

            二、const 指針和指向const的指針
              int* u, v;== int*u; int v;
                  因為c++對類型檢查非常精細,所以,你可以把一個非const指針賦給const指針,但是不能把一個const指針賦給非cosnt指針。同樣也不能把指向const的指針賦給指向非const的指針
            int d = 1;
            const int e = 2;
            int* u = &d;//right;
            int* v = &e;//wrong
            int* w = const_cast<int*>(&e);//right
            int* y = (int*)&e;//right


            考慮下面一句代碼:
            char* cp = "howdy";
            ”howdy"是存放在字符常量區的,它是個常量,這個表達式返回的是一個const char*,但是在某些編譯器中是可以通過編譯的。但最好寫成const char* cp = "howdy";修改該字符串數組是錯誤的。如果想修改它就把它放到一個數組中char cp[] = "howdy";為什么呢?
            char cp[] = "abcd";
            cp[1] = '2';
            cout <<cp << endl;
            // legall 

            char *cp = "abcd";
            cp[1] = '2';
            cout << cp << endl;//cause runtime error

            第一種情況:"abcd"是放在字符常量區的,但是數組是放在棧上的,現在棧上有5個字節的數組,分別存放了'a','b','c','d''/0',所以你修改它沒問題。第二種:"abcd"是放在字符常量區的,cp只是一個指針,指向了這串字符串,想修改它是錯誤地。

            臨時量
              他們也需要存儲空間,并且可以構造和銷毀,但是,我們看不到它們,并且自動地成為常量。一旦計算結束,臨時變量也會不復存在,所以對臨時變量所做的任何操作都會丟失。
            傳遞指針和引用的時候一般都用const修飾。
            值拷貝:
            int i = 1;
            intconst ip = &i;
            int* ip2 = ip;//right;because you can't modify ip by ip2

            const int j = 2;
            int k = j;//right;// you can't modify j by k;

            const int m = 1;
            int* mp = &m;//wrong, you can modify m by mp, but m is a constant

            類中的const
            非static變量在構造函數的初始化列表里定義
            static const 在定義的地方初始化,enum可以實現static const作用
            class A
            {
              enum {size = 100};
            int i[size];
            };//equal to static const int size = 100;

            const對象
            const A a(1); 必須保證對象的數據成員在其聲明周期中不被改變。如何保證?用const成員函數,明確表示不改變成員數據。聲明的時候要用const 定義時也要用const,不然會被看作兩個函數。
            class X
            {
            int i;
            public:
            int f() const;
            };

            int X::f() const 
            {
             return i;
            }

            按位const和按邏輯const
            按位是指這塊內存中的每個字節都是不能改變的。
            按邏輯是指,可以在某個成員函數中修改成員變量。兩種方法:

            class X
            {
              int i;
            public:
              void f() const;
            }

            void X::f() const
            {
              (const_cast<X*>(this))->i++;//right
            }

            void X::f(const X* this)
            {
               this->i++;//illegal
              (const_cast<X*>(this))->i++;//right
            }

            //////////another method//////////
            class X
            {
              mutable int i;
            public:
              void f() const;
            }

            void X::f() const
            {
              i++;//right
            }
            posted on 2012-05-30 16:12 Dino-Tech 閱讀(165) 評論(0)  編輯 收藏 引用
            久久久久久人妻无码| 亚洲色大成网站www久久九| 久久经典免费视频| 伊人热热久久原色播放www| 国产精品久久久久久久久软件| 亚洲精品乱码久久久久久蜜桃图片| 精品熟女少妇a∨免费久久| 久久亚洲精品无码播放| 久久久久无码精品国产| 99久久99久久精品国产片| 99精品国产综合久久久久五月天| 亚洲va中文字幕无码久久不卡| 久久精品www| 伊人久久大香线蕉亚洲五月天| 91精品国产高清久久久久久io| 中文字幕无码精品亚洲资源网久久| 国产精品久久久久jk制服| 久久性生大片免费观看性| 久久无码人妻一区二区三区午夜| 99久久99久久精品国产片果冻| 亚洲国产精品一区二区久久| 亚洲精品tv久久久久久久久| 亚洲成av人片不卡无码久久 | 久久中文字幕人妻丝袜| 久久精品国产久精国产| 婷婷久久久亚洲欧洲日产国码AV| 久久人妻少妇嫩草AV蜜桃| 久久这里只精品国产99热| 久久精品国产亚洲av日韩| 久久精品中文字幕大胸| 大美女久久久久久j久久| 国产国产成人精品久久| 国内精品九九久久精品 | 久久受www免费人成_看片中文| 欧美久久综合性欧美| 国产V亚洲V天堂无码久久久| 久久久国产视频| 亚洲色大成网站www久久九| 亚洲综合日韩久久成人AV| 精品久久久久久中文字幕大豆网| 久久久噜噜噜久久熟女AA片|