C++中Static作用和使用方法

來源:互聯網
上載者:User

1、什麼是static?
       static 是C++中很常用的修飾符,它被用來控制變數的儲存方式和可見度。

    2、為什麼要引入static?
       函數內部定義的變數,在程式執行到它的定義處時,編譯器為它在棧上分配空間,大家知道,函數在棧上分配的空間在此函數執行結束時會釋放掉,這樣就產生了一個問題: 如果想將函數中此變數的值儲存至下一次調用時,如何?? 最容易想到的方法是定義一個全域的變數,但定義為一個全域變數有許多缺點,最明顯的缺點是破壞了此變數的存取範圍(使得在此函數中定義的變數,不僅僅受此函數控制)。

    3、什麼時候用static?
       需要一個資料對象為整個類而非某個物件服務,同時又力求不破壞類的封裝性,即要求此成員隱藏在類的內部,對外不可見。

    4、static的內部機制:
       待用資料成員要在程式一開始運行時就必須存在。因為函數在程式運行中被調用,所以待用資料成員不能在任何函數內分配空間和初始化。
       這樣,它的空間分配有三個可能的地方,一是作為類的外部介面的標頭檔,那裡有類聲明;二是類定義的內部實現,那裡有類的成員函數定義;三是應用程式的main()函數前的全域資料聲明和定義處。
      待用資料成員要實際地分配空間,故不能在類的聲明中定義(只能聲明資料成員)。類聲明只聲明一個類的“尺寸和規格”,並不進行實際的記憶體配置,所以在類聲明中寫成定義是錯誤的。它也不能在標頭檔中類聲明的外部定義,因為那會造成在多個使用該類的源檔案中,對其重複定義。
      static被引入以告知編譯器,將變數儲存在程式的靜態儲存區而非棧上空間,靜態
資料成員按定義出現的先後順序依次初始化,注意靜態成員嵌套時,要保證所嵌套的成員已經初始化了。消除時的順序是初始化的反順序。

    5、static的優勢:
       可以節省記憶體,因為它是所有對象所公有的,因此,對多個對象來說,待用資料成員只儲存一處,供所有對象共用。待用資料成員的值對每個對象都是一樣,但它的值是可以更新的。只要對待用資料成員的值更新一次,保證所有對象存取更新後的相同的值,這樣可以提高時間效率。

    6、引用待用資料成員時,採用如下格式:
         <類名>::<靜態成員名>
    如果待用資料成員的存取權限允許的話(即public的成員),可在程式中,按上述格式
來引用待用資料成員。

    7、注意事項:
      (1)類的靜態成員函數是屬於整個類而非類的對象,所以它沒有this指標,這就導致
了它僅能訪問類的待用資料和靜態成員函數。
      (2)不能將靜態成員函數定義為虛函數。
      (3)由於靜態成員聲明於類中,操作於其外,所以對其取地址操作,就多少有些特殊
,變數地址是指向其資料類型的指標 ,函數地址類型是一個“nonmember函數指標”。

      (4)由於靜態成員函數沒有this指標,所以就差不多等同於nonmember函數,結果就
產生了一個意想不到的好處:成為一個callback函數,使得我們得以將C++和C-based X W
indow系統結合,同時也成功的應用於線程函數身上。
      (5)static並沒有增加程式的時空開銷,相反她還縮短了子類對父類靜態成員的訪問
時間,節省了子類的記憶體空間。
      (6)待用資料成員在<定義或說明>時前面加關鍵字static。
      (7)待用資料成員是靜態儲存的,所以必須對它進行初始化。
      (8)靜態成員初始化與一般資料成員初始化不同:
      初始化在類體外進行,而前面不加static,以免與一般靜態變數或對象相混淆;
      初始化時不加該成員的存取權限控制符private,public等;
           初始化時使用範圍運算子來標明它所屬類;
           所以我們得出待用資料成員初始化的格式:
         <資料類型><類名>::<待用資料成員名>=<值>
      (9)為了防止父類的影響,可以在子類定義一個與父類相同的靜態變數,以屏蔽父類的影響。這裡有一點需要注意:我們說靜態成員為父類和子類共用,但我們有重複定義了靜態成員,這會不會引起錯誤呢?不會,我們的編譯器採用了一種絕妙的手法:name-mangling 用以產生唯一的標誌。

待用資料成員

  在類中,靜態成員可以實現多個對象之間的資料共用,並且使用待用資料成員還不會破壞隱藏的原則,即保證了安全性。因此,靜態成員是類的所有對象中共用的成員,而不是某個對象的成員。

  使用待用資料成員可以節省記憶體,因為它是所有對象所公有的,因此,對多個對象來說,待用資料成員只儲存一處,供所有對象共用。待用資料成員的值對每個對象都是一樣,但它的值是可以更新的。只要對待用資料成員的值更新一次,保證所有對象存取更新後的相同的值,這樣可以提高時間效率。

  待用資料成員的使用方法和注意事項如下:

  1、待用資料成員在定義或說明時前面加關鍵字static。

  2、靜態成員初始化與一般資料成員初始化不同。待用資料成員初始化的格式如下:

    <資料類型><類名>::<待用資料成員名>=<值>

  這表明:

        (1) 初始化在類體外進行,而前面不加static,以免與一般靜態變數或對象相混淆。

  (2) 初始化時不加該成員的存取權限控制符private,public等。

  (3) 初始化時使用範圍運算子來標明它所屬類,因此,待用資料成員是類的成員,而不是對象的成員。

  3、待用資料成員是靜態儲存的,它是靜態生存期,必須對它進行初始化。

  4、引用待用資料成員時,採用如下格式:

   <類名>::<靜態成員名>

  如果待用資料成員的存取權限允許的話(即public的成員),可在程式中,按上述格式來引用待用資料成員。

靜態成員函數

  靜態成員函數和待用資料成員一樣,它們都屬於類的靜態成員,它們都不是對象成員。因此,對靜態成員的引用不需要用對象名。

  在靜態成員函數的實現中不能直接引用類中說明的非靜態成員,可以引用類中說明的靜態成員。如果靜態成員函數中要引用非靜態成員時,可通過對象來引用。

下面看一個例子:

#include <iostream.h>class  Point{  public:  void output()  {  }  static void init()  {   }};void main( void ){   Point pt;   pt.init();   pt.output(); }

這樣編譯是不會有任何錯誤的。
下面這樣看

#include <iostream.h>class  Point{    public:    void output()    {     }     static void init()     {      }};void main( void ){     Point::output();}

這樣編譯會處錯,錯誤資訊:illegal call of non-static member function,為什嗎?
因為在沒有執行個體化一個類的具體對象時,類是沒有被分配記憶體空間的。
好的再看看下面的例子:

#include <iostream.h>class  Point{ public: void output() {  } static void init() {  }};void main( void ){   Point::init();}

這時編譯就不會有錯誤,因為在類的定義時,它待用資料和成員函數就有了它的記憶體區,它不屬於類的任何一個具體對象。
好的再看看下面的例子:

#include <iostream.h>class  Point{ public: void output() {  } static void init() {    x = 0;    y = 0; } private:   int x;   int y;};void main( void ){   Point::init();}

編譯出錯:
illegal reference to data member 'Point::x' in a static member function
illegal reference to data member 'Point::y' in a static member function
在一個靜態成員函數裡錯誤的引用了資料成員,
還是那個問題,靜態成員(函數),不屬於任何一個具體的對象,那麼在類的具體對象聲明之前就已經有了記憶體區,
而現在非待用資料成員還沒有分配記憶體空間,那麼這裡調用就錯誤了,就好像沒有聲明一個變數卻提前使用它一樣。
也就是說在靜態成員函數中不能引用非靜態成員變數。
好的再看看下面的例子:

#include <iostream.h>class  Point{public: void output() {   x = 0;   y = 0;   init();  } static void init() { }private:  int x;  int y;};void main( void ){  Point::init();}

好的,這樣就不會有任何錯誤。這最終還是一個記憶體模型的問題,
任何變數在記憶體中有了自己的空間後,在其他地方才能被調用,否則就會出錯。
好的再看看下面的例子:

#include <iostream.h>class  Point{public: void output() { } static void init() {   x = 0;   y = 0; }private:  static int x;  static int y;};void main( void ){  Point::init();}

編譯:
Linking...
test.obj : error LNK2001: unresolved external symbol "private: static int Point::y"
test.obj : error LNK2001: unresolved external symbol "private: static int Point::x"
Debug/Test.exe : fatal error LNK1120: 2 unresolved externals
執行 link.exe 時出錯.
可以看到編譯沒有錯誤,串連錯誤,這又是為什麼呢?
這是因為靜態成員變數要進行初始化,可以這樣:

#include <iostream.h>class  Point{public: void output() { } static void init() {   x = 0;   y = 0; }private:  static int x;  static int y;};int Point::x = 0;int Point::y = 0;void main( void ){  Point::init();}

在靜態成員資料變數初始化之後就不會出現編譯錯誤了。
再看看下面的代碼:

#include <iostream.h>class  Point{public: void output() { } static void init() {   x = 0;   y = 0; }private:  static int x;  static int y;};void main( void ){}

編譯沒有錯誤,為什嗎?
即使他們沒有初始化,因為我們沒有訪問x,y,所以編譯不會出錯。 

C++會區分兩種類型的成員函數:靜態成員函數和非靜態成員函數。這兩者之間的一個重大區別是,靜態成員函數不接受隱含的this自變數。所以,它就無法訪問自己類的非靜態成員。

在某些條件下,比如說在使用諸如pthread(它不支援類)此類的多線程庫時,就必須使用靜態成員函數,因為其地址同C語言函數的地址相容。這種銅限制就迫使程式員要利用各種解決辦法才能夠從靜態成員函數訪問到非待用資料成員。

第一個解決辦法是聲明類的所有資料成員都是靜態。運用這種方式的話,靜態成員函數就能夠直接地訪問它們,例如:

class Singleton{ public:   static Singleton * instance(); private:   Singleton * p;   static Lock lock;};Singleton * Singleton::instance(){  lock.getlock(); // fine, lock is static  if (!p)  p=new Singleton;  lock.unlock();  return p;}

 

這種解決方案不適用於需要使用非待用資料成員的類。

訪問非待用資料成員

將參照傳遞給需要考量的對象能夠讓靜態成員函數訪問到對象的非待用資料:

class A{public:  static void func(A & obj);  int getval() const; //non-static member functionprivate:  intval;};

 

靜態成員函數func()會使用參照obj來訪問非靜態成員val。

void A::func(A & obj){   int n = obj.getval();}

 

將一個參照或者指標作為靜態成員函數的自變數傳遞,就是在模仿自動傳遞非靜態成員函數裡this自變數這一行為。

 

轉自:http://www.2cto.com/kf/201110/109141.html

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.