• <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 閱讀(164) 評論(0)  編輯 收藏 引用
            国产精品久久久久9999| 亚洲av成人无码久久精品| 91精品国产高清久久久久久io| 亚洲精品乱码久久久久久蜜桃图片 | 久久AV高清无码| 久久久久人妻一区二区三区vr| 国内精品久久九九国产精品| 国内精品久久久久久久影视麻豆| 少妇久久久久久被弄到高潮| 久久精品视频免费| 精品久久久久中文字幕一区| 欧美精品九九99久久在观看| 97久久精品人妻人人搡人人玩| 99久久人人爽亚洲精品美女| 一本色道久久综合亚洲精品| 91精品国产91久久久久久蜜臀| 久久伊人五月丁香狠狠色| 精品国际久久久久999波多野| 狠狠色丁香久久婷婷综合_中| 国产成人久久精品区一区二区| 久久午夜福利无码1000合集| 精品久久久久中文字| 精品国产乱码久久久久久1区2区 | 中文字幕日本人妻久久久免费| 国产日产久久高清欧美一区| 久久99热这里只有精品国产| 久久综合狠狠综合久久97色| 青青青国产精品国产精品久久久久| 午夜久久久久久禁播电影| 色综合久久中文综合网| 无码国内精品久久人妻蜜桃| 亚洲乱码日产精品a级毛片久久| 久久久久亚洲AV综合波多野结衣 | 久久91亚洲人成电影网站| 国内精品伊人久久久久妇| 久久婷婷人人澡人人| 精品无码久久久久久久久久| 久久久精品国产亚洲成人满18免费网站| 久久精品草草草| 青青青青久久精品国产h| 国产精品久久久天天影视|