• <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年5月>
            293012345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789

            常用鏈接

            留言簿

            隨筆分類

            隨筆檔案

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            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)  編輯 收藏 引用
            国产免费福利体检区久久| 精品久久久久久成人AV| MM131亚洲国产美女久久| 色婷婷综合久久久中文字幕| 久久99精品久久久久久久不卡| 久久久久99精品成人片直播| 一级做a爰片久久毛片看看| 精品国产乱码久久久久久人妻| 色狠狠久久AV五月综合| 久久er热视频在这里精品| 伊人色综合久久天天人守人婷 | 一级做a爰片久久毛片毛片| 精品久久久久成人码免费动漫| 狠狠色丁香婷婷综合久久来| 精品久久久久久无码中文野结衣| 精品无码久久久久国产| 国产真实乱对白精彩久久| 色欲久久久天天天综合网| 青青青青久久精品国产h久久精品五福影院1421 | 亚洲AV无码1区2区久久| 久久精品成人| 人人狠狠综合88综合久久| 成人久久综合网| 91精品国产色综合久久| 国产精品久久久久久久app| 国产欧美久久一区二区| 日本WV一本一道久久香蕉| 中文精品久久久久国产网址 | 一本一道久久精品综合| 麻豆成人久久精品二区三区免费| 日本高清无卡码一区二区久久| 欧美精品一区二区精品久久| 久久国产精品77777| 久久伊人五月丁香狠狠色| 狠狠色丁香婷婷久久综合五月 | 亚洲精品乱码久久久久久蜜桃图片| 亚洲欧洲中文日韩久久AV乱码| 亚洲日韩欧美一区久久久久我 | 热久久最新网站获取| 久久99这里只有精品国产| 国产精品日韩欧美久久综合|