久久国产成人av_抖音国产毛片_a片网站免费观看_A片无码播放手机在线观看,色五月在线观看,亚洲精品m在线观看,女人自慰的免费网址,悠悠在线观看精品视频,一级日本片免费的,亚洲精品久,国产精品成人久久久久久久

分享

const與#define的優(yōu)缺點?

 e3399 2011-06-07

1.define由預(yù)處理程序處理,,const由編譯程序處理

2.#define不分內(nèi)存,,因為它是預(yù)編譯指令,編譯前進行了宏替換,。const 不一定,?某種說法,Const常量是占有內(nèi)存的被凍結(jié)了的變量

3.const定義常量是有數(shù)據(jù)類型的,,這樣const定義的常量編譯器可以對其進行數(shù)據(jù)靜態(tài)類型安全檢查,,而#define宏定義的常量卻只是進行簡單的字符替換,沒有類型安全檢查,,且有時還會產(chǎn)生邊際效應(yīng)

4.有些調(diào)試程序可對const進行調(diào)試,,但不對#define進行調(diào)試。

5.const在編譯期間會計算其值,define不會

6 當定義局部變量時,,const作用域僅限于定義局部變量的函數(shù)體內(nèi),。但用#define時其作用域不僅限于定義局部變量的函數(shù)體內(nèi),,而是從定義點到整個程序的結(jié)束點,。但也可以用#undef取消其定義從而限定其作用域范圍。只用const定義常量,,并不能起到其強大的作用,。const還可修飾函數(shù)形式參數(shù)、返回值和類的成員函數(shù)等,。從而提高函數(shù)的健壯性,。因為const修飾的東西能受到c/c++的靜態(tài)類型安全檢查機制的強制保護,防止意外的修改,。

 

二,、constc++

       看了傳遞方式后我們繼續(xù)來談“const只能用于修飾輸入?yún)?shù)的情況,。

          當輸入?yún)?shù)用值傳遞方式時,我們不需要加const修飾,,因為用值傳遞時,,函數(shù)將自動用實際參數(shù)的拷貝初始化形式參數(shù),當在函數(shù)體內(nèi)改變形式參數(shù)時,,改變的也只是棧上的拷貝而不是實際參數(shù),。
         
但要注意的是,當輸入?yún)?shù)為ADT/UDT(用戶自定義類型和抽象數(shù)據(jù)類型)時,,應(yīng)該將值傳遞改為“const &傳遞,,目的可以提高效率。
         
例如:
             void fun(A a);//
效率底,。函數(shù)體內(nèi)產(chǎn)生A類型的臨時對象用于復(fù)制參數(shù) a,,但是臨時對象的
                           //
構(gòu)造、復(fù)制,、析構(gòu)過程都將消耗時間,。
             void fun(A const &a);//
提高效率。用引用傳遞不需要產(chǎn)生臨時對象,,省了臨時對象的
                                  //
構(gòu)造,、復(fù)制、析構(gòu)過程消耗的時間,。但光用引用有可能改變a,所以加const


         
當輸入?yún)?shù)用指針傳遞方式時,,加const修飾可防止意外修改指針指向的內(nèi)存單元,起到保護作用,。
         
例如:
             void funstrcopy(char *strdest,const char *strsrc)//
任何改變strsrc指向的內(nèi)存單元,,
                                                              //
編譯器都將報錯
            
些時保護了指針的內(nèi)存單元,也可以保護指針本身,,防止其地址改變,。
         
例如:
            void funstrcopy(char *strdest,const char *const strsrc)

(3)const修飾函數(shù)的返回值

      如給指針傳遞的函數(shù)返回值加const,則返回值不能被直接修改,,且該返回值只能被賦值給加const修飾的同類型指針,。
     
例如:
         const char *GetChar(void){};
      
賦值 char *ch = GetChar();//錯誤     const char *ch = GetChar();//正確

(4)const修飾類的成員函數(shù)(函數(shù)定義體)

      任何不會修改數(shù)據(jù)成員的函數(shù)都應(yīng)用const修飾,這樣當不小心修改了數(shù)據(jù)成員或調(diào)用了非const成員函數(shù)時,,編譯器都會報錯,。
      const
修飾類的成員函數(shù)形式為:int GetCount(void)   const;
5)用傳引用給const取代傳值
缺省情況下,C++ 以傳值方式將對象傳入或傳出函數(shù)(這是一個從 C 繼承來的特性),。除非你特別指定其它方式,,否則函數(shù)的參數(shù)就會以實際參數(shù)(actual argument)的拷貝進行初始化,而函數(shù)的調(diào)用者會收到函數(shù)返回值的一個拷貝,。這個拷貝由對象的拷貝構(gòu)造函數(shù)生成,。這就使得傳值(pass-by-value)成為一個代價不菲的操作,。例如,考慮下面這個類層級結(jié)構(gòu):

class Person {
 public:
  Person(); // parameters omitted for simplicity
  virtual ~Person(); // see Item 7 for why this is virtual
  ...

 private:
  std::string name;
  std::string address;
};

class Student: public Person {
 public:
  Student(); // parameters again omitted
  ~Student();
  ...

 private:
  std::string schoolName;
  std::string schoolAddress;
};

  現(xiàn)在,,考慮以下代碼,,在此我們調(diào)用一個函數(shù)—— validateStudent,它得到一個 Student 參數(shù)(以傳值的方式),,并返回它是否驗證有效的結(jié)果:

bool validateStudent(Student s); // function taking a Student
// by value

Student plato; // Plato studied under Socrates

bool platoIsOK = validateStudent(plato); // call the function

  當這個函數(shù)被調(diào)用時會發(fā)生什么呢,?

  很明顯,Student 的拷貝構(gòu)造函數(shù)被調(diào)用,,用 plato 來初始化參數(shù) s,。同樣明顯的是,當 validateStudent 返回時,,s 就會被銷毀,。所以這個函數(shù)的參數(shù)傳遞代價是一次 Student 的拷貝構(gòu)造函數(shù)的調(diào)用和一次 Student 的析構(gòu)函數(shù)的調(diào)用。

  但這還不是全部,。一個 Student 對象內(nèi)部包含兩個 string 對象,,所以每次你構(gòu)造一個 Student 對象的時候,你也必須構(gòu)造兩個 string 對象,。一個 Student 對象還要從一個 Person 對象繼承,,所以每次你構(gòu)造一個 Student 對象的時候,你也必須構(gòu)造一個 Person 對象,。一個 Person 對象內(nèi)部又包含兩個額外的 string 對象,,所以每個 Person 的構(gòu)造也承擔(dān)著另外兩個 string 的構(gòu)造。最終,,以傳值方式傳遞一個 Student 對象的后果就是引起一次 Student 的拷貝構(gòu)造函數(shù)的調(diào)用,,一次 Person 的拷貝構(gòu)造函數(shù)的調(diào)用,以及四次 string 的拷貝構(gòu)造函數(shù)調(diào)用,。當 Student 對象的拷貝被銷毀時,,每一個構(gòu)造函數(shù)的調(diào)用都對應(yīng)一個析構(gòu)函數(shù)的調(diào)用,所以以傳值方式傳遞一個 Student 的全部代價是六個構(gòu)造函數(shù)和六個析構(gòu)函數(shù),!

  好了,,這是正確的和值得的行為。畢竟,,你希望你的全部對象都得到可靠的初始化和銷毀,。盡管如此,,如果有一種辦法可以繞過所有這些構(gòu)造和析構(gòu)過程,,應(yīng)該變得更好,這就是:傳引用給 constpass by reference-to-const):

bool validateStudent(const Student& s);

  這樣做非常有效:沒有任何構(gòu)造函數(shù)和析構(gòu)函數(shù)被調(diào)用,,因為沒有新的對象被構(gòu)造,。被修改的參數(shù)聲明中的 const 是非常重要的,。 validateStudent 的最初版本接受一個 Student 值參數(shù),所以調(diào)用者知道它們屏蔽了函數(shù)對它們傳入的 Student 的任何可能的改變,;validateStudent 也只能改變它的一個拷貝?,F(xiàn)在 Student 以引用方式傳遞,同時將它聲明為 const 是必要的,,否則調(diào)用者必然擔(dān)心 validateStudent 改變了它們傳入的 Student,。

  以傳引用方式傳遞參數(shù)還可以避免切斷問題(slicing problem)。當一個派生類對象作為一個基類對象被傳遞(傳值方式),,基類的拷貝構(gòu)造函數(shù)被調(diào)用,,而那些使得對象的行為像一個派生類對象的特殊特性被切斷了。你只剩下一個純粹的基類對象——這沒什么可吃驚的,,因為是一個基類的構(gòu)造函數(shù)創(chuàng)建了它,。這幾乎絕不是你希望的。例如,,假設(shè)你在一組實現(xiàn)一個圖形窗口系統(tǒng)的類上工作:

class Window {
 public:
  ...
  std::string name() const; // return name of window
  virtual void display() const; // draw window and contents
};

class WindowWithScrollBars: public Window {
 public:
  ...
  virtual void display() const;
};

  所有 Window 對象都有一個名字,,你能通過 name 函數(shù)得到它,而且所有的窗口都可以顯示,,你可一個通過調(diào)用 display 函數(shù)來做到這一點,。display virtual 的事實清楚地告訴你:一個純粹的基類的 Window 對象的顯示方法有可能不同于專門的 WindowWithScrollBars 對象的顯示方法。

  現(xiàn)在,,假設(shè)你想寫一個函數(shù)打印出一個窗口的名字,,并隨后顯示這個窗口。以下這個函數(shù)的寫法是錯誤的:

void printNameAndDisplay(Window w) // incorrect! parameter
{
 
 // may be sliced!
 std::cout << w.name();
 w.display();
}

  考慮當你用一個 WindowWithScrollBars 對象調(diào)用這個函數(shù)時會發(fā)生什么:

WindowWithScrollBars wwsb;

printNameAndDisplay(wwsb);

  參數(shù) w 將被作為一個 Window 對象構(gòu)造——它是被傳值的,,記得嗎,?而且使 wwsb 表現(xiàn)得像一個 WindowWithScrollBars 對象的特殊信息都被切斷了。在 printNameAndDisplay 中,,全然不顧傳遞給函數(shù)的那個對象的類型,,w 將始終表現(xiàn)得像一個 Window 類的對象(因為它就是一個 Window 類的對象)。特別是,,在 printNameAndDisplay 中調(diào)用 display 將總是調(diào)用 Window::display,,絕不會是 WindowWithScrollBars::display

  繞過切斷問題的方法就是以傳引用給 const 的方式傳遞 w

void printNameAndDisplay(const Window& w) // fine, parameter won’t
{
 // be sliced
 std::cout << w.name();
 w.display();
}

  現(xiàn)在 w 將表現(xiàn)得像實際傳入的那種窗口,。

  如果你掀開編譯器的蓋頭偷看一下,,你會發(fā)現(xiàn)用指針實現(xiàn)引用是非常典型的做法,所以以引用傳遞某物實際上通常意味著傳遞一個指針,。由此可以得出結(jié)論,,如果你有一個內(nèi)建類型的對象(例如,一個 int),,以傳值方式傳遞它常常比傳引用方式更高效,。那么,,對于內(nèi)建類型,當你需要在傳值和傳引用給 const 之間做一個選擇時,,沒有道理不選擇傳值,。同樣的建議也適用于 STL 中的迭代器(iterators)和函數(shù)對象(function objects),因為,,作為慣例,,它們就是為傳值設(shè)計的。迭代器(iterators)和函數(shù)對象(function objects)的實現(xiàn)有責(zé)任保證拷貝的高效并且不受切斷問題的影響,。(這是一個規(guī)則如何變化,,依賴于你使用 C++ 的哪一個部分的實例。)

  內(nèi)建類型很小,,所以有人就斷定所有的小類型都是傳值的上等候選者,,即使它們是用戶定義的。這樣的推論是不可靠的,。僅僅因為一個對象小,,并不意味著調(diào)用它的拷貝構(gòu)造函數(shù)就是廉價的。很多對象——大多數(shù) STL 容器也在其中——容納的和指針一樣,,但是拷貝這樣的對象必須同時拷貝它們指向的每一樣?xùn)|西,。那可能是非常昂貴的。

  即使當一個小對象有一個廉價的拷貝構(gòu)造函數(shù),,也會存在性能問題,。一些編譯器對內(nèi)建類型和用戶定義類型并不一視同仁,即使他們有同樣的底層表示,。例如,,一些編譯器拒絕將僅由一個 double 組成的對象放入一個寄存器中,即使在常規(guī)上它們非常愿意將一個純粹的 double 放入那里,。如果發(fā)生了這種事情,,你以傳引用方式傳遞這樣的對象更好一些,因為編譯器理所當然會將一個指針(引用的實現(xiàn))放入寄存器,。

  小的用戶定義類型不一定是傳值的上等候選者的另一個原因是:作為用戶定義類型,,它的大小常常變化。一個現(xiàn)在較小的類型在將來版本中可能變得更大,,因為它的內(nèi)部實現(xiàn)可能會變化,。甚至當你換了一個不同的 C++ 實現(xiàn)時,事情都可能會變化,。例如,,就在我這樣寫的時候,一些標準庫的 string 類型的實現(xiàn)的大小就是另外一些實現(xiàn)的七倍。

  通常情況下,,你能合理地假設(shè)傳值廉價的類型僅有內(nèi)建類型及 STL 中的迭代器和函數(shù)對象類型。對其他任何類型,,請遵循本 Item 的建議,,并用傳引用給 const 取代傳值。

  Things to Remember

  ·用傳引用給 const 取代傳值,。典型情況下它更高效而且可以避免切斷問題,。

  ·這條規(guī)則并不適用于內(nèi)建類型及 STL 中的迭代器和函數(shù)對象類型。對于它們,,傳值通常更合適,。

 

原文來源:http://dev./course/3_program/c++/cppjs/20091014/178760.html

    本站是提供個人知識管理的網(wǎng)絡(luò)存儲空間,所有內(nèi)容均由用戶發(fā)布,,不代表本站觀點,。請注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購買等信息,,謹防詐騙,。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點擊一鍵舉報,。
    轉(zhuǎn)藏 分享 獻花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多