靜態(tài)函數(shù)
用static聲明的函數(shù)是靜態(tài)函數(shù)。靜態(tài)函數(shù)可以分為全局靜態(tài)函數(shù)和類的靜態(tài)成員函數(shù)。
Static關(guān)鍵字
在類中,用static聲明的成員變量為靜態(tài)成員變量,它為該類的公用變量,在第一次使用時(shí)被初始化,對(duì)于該類的所有對(duì)象來說,static成員變量只有一份。
用static聲明的方法是靜態(tài)方法,在調(diào)用該方法時(shí),不會(huì)將對(duì)象的引用傳遞給它,所以在static方法中不可訪問非static的成員。
靜態(tài)方法不再是針對(duì)于某個(gè)對(duì)象調(diào)用,所以不能訪問非靜態(tài)成員。
可以通過對(duì)象引用或類名(不需要實(shí)例化)訪問靜態(tài)成員
C++類靜態(tài)數(shù)據(jù)成員與類靜態(tài)成員函數(shù)
函數(shù)調(diào)用的結(jié)果不會(huì)訪問或者修改任何對(duì)象(非static)數(shù)據(jù)成員,這樣的成員聲明為靜態(tài)成員函數(shù)比較好。且如果static int func(....)不是出現(xiàn)在類中,則它不是一個(gè)靜態(tài)成員函數(shù),只是一個(gè)普通的全局函數(shù),只不過由于static的限制,它只能在文件所在的編譯單位內(nèi)使用,不能在其它編譯單位內(nèi)使用。
靜態(tài)成員函數(shù)的聲明除了在類體的函數(shù)聲明前加上關(guān)鍵字static,以及不能聲明為const或者volatile之外,與非靜態(tài)成員函數(shù)相同。出現(xiàn)在類體之外的函數(shù)定義不能制定關(guān)鍵字static。
靜態(tài)成員函數(shù)沒有this指針。
在沒有講述本章內(nèi)容之前如果我們想要在一個(gè)范圍內(nèi)共享某一個(gè)數(shù)據(jù),那么我們會(huì)設(shè)立全局對(duì)象,但面向?qū)ο蟮某绦蚴怯蓪?duì)象構(gòu)成的,我們?nèi)绾尾拍茉陬惙秶鷥?nèi)共享數(shù)據(jù)呢?
這個(gè)問題便是本章的重點(diǎn):聲明為static的類成員或者成員函數(shù)便能在類的范圍內(nèi)共同享,我們把這樣的成員稱做靜態(tài)成員和靜態(tài)成員函數(shù)。
下面我們用幾個(gè)實(shí)例來說明這個(gè)問題,類的成員需要保護(hù),通常情況下為了不違背類的封裝特性,我們是把類成員設(shè)置為protected(保護(hù)狀態(tài))的,但是我們?yōu)榱撕喕a,使要說明的問題更為直觀,更容易理解,我們?cè)诖颂幎荚O(shè)置為public。
以下程序我們來做一個(gè)模擬訪問的例子,在程序中,每建立一個(gè)對(duì)象我們?cè)O(shè)置的類靜態(tài)成員變自動(dòng)加一,代碼如下:
#include <iostream>
using namespace std;
class Internet
{
public:
Internet(char *name,char *address)
{
strcpy(Internet::name,name);
strcpy(Internet::address,address);
count++;
}
static void Internet::Sc()//靜態(tài)成員函數(shù)
{
cout<<count<<endl;
}
Internet &Rq();
public:
char name[20];
char address[20];
static int count;//這里如果寫成static int count=0;就是錯(cuò)誤的
};
Internet& Internet::Rq()//返回引用的成員函數(shù)
{
return *this;
}
int Internet::count = 0;//靜態(tài)成員的初始化
void vist()
{
Internet a1("中國軟件開發(fā)實(shí)驗(yàn)室","www.cndev-lab.com");
Internet a2("中國軟件開發(fā)實(shí)驗(yàn)室","www.cndev-lab.com");
}
void fn(Internet &s)
{
cout<<s.Rq().count;
}
void main()
{
cout<<Internet::count<<endl;//靜態(tài)成員值的輸出
vist();
Internet::Sc();//靜態(tài)成員函數(shù)的調(diào)用
Internet b("中國軟件開發(fā)實(shí)驗(yàn)室","www.cndev-lab.com");
Internet::Sc();
fn(b);
cin.get();
}
上面代碼我們用了幾種常用的方式建立對(duì)象,當(dāng)建立新對(duì)象并調(diào)用其構(gòu)造函數(shù)的時(shí)候,靜態(tài)成員cout便運(yùn)行加1操作,靜態(tài)成員的初始化應(yīng)該在主函數(shù)調(diào)用之前,并且不能在類的聲明中出現(xiàn),通過運(yùn)行過程的觀察我們發(fā)現(xiàn),靜態(tài)成員count的狀態(tài)并不會(huì)隨著一個(gè)新的對(duì)象的新建而重新定義,盡而我們了解到類的靜態(tài)成員是屬于類的而不是屬于哪一個(gè)對(duì)象的,所以靜態(tài)成員的使用應(yīng)該是類名稱加域區(qū)分符加成員名稱的,在上面的代碼中就是Internet::count,雖然我們?nèi)匀豢梢允褂脤?duì)象名加點(diǎn)操作符號(hào)加成員名稱的方式使用,但是不推薦的,靜態(tài)類成員的特性就是屬于類而不專屬于某一個(gè)對(duì)象。
靜態(tài)成員函數(shù)的特性類似于靜態(tài)成員的使用,同樣與對(duì)象無關(guān),調(diào)用方法為類名稱加域區(qū)分符加成員函數(shù)名稱,在上面的代碼中就是Internet::Sc();靜態(tài)成員函數(shù)由于與對(duì)象無關(guān)系,所以在其中是不能對(duì)類的普通成員進(jìn)行直接操作的。
如果上面的 static void Internet::Sc()修改成為:
static void Internet::Sc()//靜態(tài)成員函數(shù)
{
cout<<name<<endl;//錯(cuò)誤
cout<<count<<endl;
}
靜態(tài)成員函數(shù)與普通成員函數(shù)的差別就在于缺少this指針,沒有這個(gè)this指針自然也就無從知道name是哪一個(gè)對(duì)象的成員了。
根據(jù)類靜態(tài)成員的特性我們可以簡單歸納出幾點(diǎn),靜態(tài)成員的使用范圍:
1.用來保存對(duì)象的個(gè)數(shù)。
2.作為一個(gè)標(biāo)記,標(biāo)記一些動(dòng)作是否發(fā)生,比如:文件的打開狀態(tài),打印機(jī)的使用狀態(tài),等等。
3.存儲(chǔ)鏈表的第一個(gè)或者最后一個(gè)成員的內(nèi)存地址。
為了做一些必要的練習(xí),深入的掌握靜態(tài)對(duì)象的存在的意義,我們以前面的結(jié)構(gòu)體的教程為基礎(chǔ),用類的方式描述一個(gè)線性鏈表,用于存儲(chǔ)若干學(xué)生的姓名,代碼如下:
#include <iostream>
using namespace std;
class Student
{
public:
Student (char *name);
~Student();
public:
char name[30];
Student *next;
static Student *point;
};
Student::Student(char *name)
{
strcpy(Student::name,name);
this->next=point;
point=this;
}
Student::~Student ()//析構(gòu)過程就是節(jié)點(diǎn)的脫離過程
{
cout<<"析構(gòu):"<<name<<endl;
if(point==this)
{
point=this->next;
cin.get();
return;
}
for(Student *ps=point;ps;ps=ps->next)
{
if(ps->next==this)
{
cout<<ps->next<<""<<this->next<<endl;
ps->next=next;//=next也可以寫成this->next;
cin.get();
return;
}
}
cin.get();
}
Student* Student::point=NULL;
void main()
{
Student *c = new Student("marry");
Student a("colin");
Student b("jamesji");
delete c;
Student *fp=Student::point;
while(fp!=NULL)
{
cout<<fp->name<<endl;
fp=fp->next;
}
cin.get();
}
從上面的代碼來看,原來單純結(jié)構(gòu)化編程需要的一個(gè)鏈表進(jìn)入全局指針在這里被類的靜態(tài)成員指針?biāo)娲?span lang="EN-US">(類的靜態(tài)成員完全可以替代全局變量),這個(gè)例子的理解重點(diǎn)主要是要注意觀察類成員的析構(gòu)順序,通過對(duì)析構(gòu)順序的理解,使用析構(gòu)函數(shù)來進(jìn)行節(jié)點(diǎn)的脫鏈操作。
為什么虛函數(shù)必須是非靜態(tài)成員函數(shù)
如果定義為虛函數(shù),那么它就是動(dòng)態(tài)綁定的,也就是在派生類中可以被覆蓋的,這與靜態(tài)成員函數(shù)的定義本身就是相矛盾的。
==
主要有兩個(gè)作用:
1、管理靜態(tài)數(shù)據(jù)成員;
2、提供類范圍的功能,即不需要對(duì)象來實(shí)現(xiàn)的功能。
比如Symbian中的NewL/LC方法就是static的
==
使用static關(guān)鍵字聲明的函數(shù)成員是靜態(tài)的,靜態(tài)成員函數(shù)同樣也屬于整個(gè)類,由同一個(gè)類的所有對(duì)象共同維護(hù),為這些對(duì)象所共享。
作為成員函數(shù),它的訪問屬性可以受到類的嚴(yán)格控制,對(duì)于公有的靜態(tài)函數(shù)成員函數(shù),可以通過類名或?qū)ο竺麃碚{(diào)用,但一般情況下建議用對(duì)象名來引用靜態(tài)函數(shù)成員(真的嗎?)。注意,一般的成員函數(shù)只能通過對(duì)象名來調(diào)用。
由于一個(gè)類的靜態(tài)成員函數(shù)只有一個(gè)拷貝,因此它訪問對(duì)象的數(shù)據(jù)何函數(shù)使受到了限制。靜態(tài)成員函數(shù)可以直接訪問該類的靜態(tài)數(shù)據(jù)成員。而訪問非靜態(tài)數(shù)據(jù)成員,必須通過參數(shù)傳遞方式得到對(duì)象名,然后通過對(duì)象名來訪問。可以看到,通過靜態(tài)函數(shù)成員訪問非靜態(tài)成員使相當(dāng)麻煩的,一般的使用中,它主要用來訪問全局變量或同一個(gè)類中的靜態(tài)數(shù)據(jù)成員,特別是和后者一起使用,達(dá)到對(duì)同一個(gè)類中對(duì)象之間共享的數(shù)據(jù)進(jìn)行維護(hù)的目的。
構(gòu)造函數(shù)不可以定義為static,看了上面,應(yīng)該可以理解原因。
注意,由于static不是函數(shù)類型中的一部分,所以在類定義之外定義靜態(tài)成員函數(shù)時(shí)不使用static,在類中定義的靜態(tài)成員函數(shù)是內(nèi)聯(lián)的。
一般來說,通過成員名限定比使用對(duì)象名訪問靜態(tài)成員要好。因?yàn)殪o態(tài)成員不是對(duì)象的成員。
靜態(tài)成員可以被繼承,這時(shí),基類對(duì)象和派生類的對(duì)象共享該靜態(tài)成員,除此之外,在類等級(jí)中對(duì)靜態(tài)成員的其他特性(例如,靜態(tài)成員在派生類中的訪問權(quán)限,在派生類中重載成員函數(shù)等)的分析與一般成員類似。
靜態(tài)成員函數(shù)不能被申明為虛函數(shù),靜態(tài)成員具有外部連接屬性,static僅有的含義是使該成員為該類的所有對(duì)象共享。
類中的任何成員函數(shù)都可以訪問靜態(tài)成員,但靜態(tài)成員函數(shù)只能通過對(duì)象名(或指向?qū)ο蟮闹羔槪┰L問該對(duì)象的非靜態(tài)成員,因?yàn)殪o態(tài)成員函數(shù)沒有this 指針。
虛函數(shù)必須是非靜態(tài)成員函數(shù)
構(gòu)造函數(shù)不可以定義為static