c++中的 static 關鍵字
注:若沒有特指是 靜態成員時,預設都是普通成員;
1 類中的普通成員
類中的成員變數 和 成員函式 是分開儲存的。其中,
1)每個物件都有獨立的成員變數;成員變數可以儲存在 棧空間、堆空間、全域性資料區;
2)所有物件共享類的成員函式;成員函式 只能儲存在 程式碼段;
2 類中的靜態成員(static)
類中的靜態成員
1、用 static關鍵字 修飾;
2、可以用 類名::成員名 訪問 靜態成員;
3、靜態成員 屬於 整個類;
4、靜態成員 是所屬類的成員,其它類不能訪問;
5、靜態成員的記憶體分配 是 唯一的;
1) 靜態成員變數
特徵:1、靜態成員變數 屬於 整個類所有;
2、靜態成員變數的生命週期不依賴任何物件;(靜態成員變數的生命週期在程式的執行期)
3、所有物件共享類的靜態成員變數;
4、可以通過 類名 直接訪問公有的靜態成員變數;
5、可以通過 物件名 訪問公有的靜態成員變數;
6、靜態成員變數 需要在類外單獨分配空間;(類內宣告、類外定義並初始化)
7、靜態成員變數 在程式內部位於全域性資料區,不計入類的記憶體計算。
原因/好處:使用靜態成員變數實現多個物件之間的資料共享不會破壞隱藏的原則,保證了安全性還可以節省記憶體。
使用方法:
1、在類的內部,使用 static 修飾普通成員變數;
2、在類的外部(全域性作用域),使用 Type ClassName::VarName = value 初始化,並申請儲存空間;
注:靜態成員變數不屬於類的任何物件,所以並不是物件建立時被定義的,所以它不能由類的建構函式初始化,一般也不能在類內初始化;
1 /* 2 靜態成員變數 只能在類的內部宣告,在類的外部(全域性區)定義和初始化; 3 */ 4 5 #include <iostream> 6 7 using namespace std; 8 9 class Test{ 10 public: 11 int GetA() const{return a;} 12 private: 13 static int a; // 靜態成員變數 14 }; 15 //int Test::a;如果這樣定義不賦予初值,則初值為零 16 int Test::a = 1; 17 18 int main(int argc, char *argv[]) 19 { 20 Test T; 21 22 cout << T.GetA() << endl; 23 24 return 0; 25 }
靜態成員變數 被類的所有物件共享,包括派生類物件;
1 #include <iostream> 2 3 using namespace std; 4 5 class Base{ 6 public: 7 static int a; // 靜態成員變數 8 }; 9 10 // int Test::a;如果這樣定義不賦予初值,則初值為零 11 int Base::a; 12 13 class Derived : public Base{ 14 15 }; 16 17 18 int main(int argc, char *argv[]) 19 { 20 Base B; 21 Derived D; 22 23 B.a++; 24 cout << B.a << endl; // 1 25 D.a++; 26 cout << D.a << endl; // 2 27 28 return 0; 29 }
靜態成員變數可以作為普通成員函式的預設形參,而普通成員變數則不可以;
1 class Test{ 2 public: 3 static int a; //靜態成員變數 4 int b; 5 void fun_1(int i = a); //正確 6 //void fun_2(int i = b); //報錯 7 };
靜態成員變數的型別 可以是所屬類的型別,而普通成員變數則不可以。普通成員變數只能宣告為 所屬類型別的 指標或引用;
1 class Test{ 2 public: 3 static Test a; //正確 4 Test b; //報錯 5 Test *pTest; //正確 6 Test &m_Test; //正確 7 static Test *pStaticObject; //正確 8 };
靜態成員變數在const函式中可以修改,而普通的成員變數是萬萬不能修改的;
1 /* 2 const修飾的是當前this指標所指向的物件是const,但是靜態成員變數不屬於任何類的物件,它被類的所有物件修改,所以this指標不修飾靜態的成員變數,所以可以更改。 3 */ 4 class Test{ 5 public: 6 static int a; 7 int b; 8 public: 9 Test():b(0){} 10 void test() const 11 { 12 a++; 13 //b++; // err // const指的是不能修改當前呼叫該函式物件的成員變數 14 } 15 }; 16 17 int Test::a;
2)靜態成員函式
特徵:1、靜態成員函式 屬於 整個類所有;
2、所有物件共享類的靜態成員函式;
2、可以通過 類名 直接訪問公有的靜態成員函式;
3、可以通過 物件名 訪問公有的靜態成員函式;
4、靜態成員函式 只能 訪問靜態成員,不能訪問 非靜態成員;
5、靜態成員函式沒有this指標,也就是說靜態成員函式不能使用修飾符(也就是函式後面的const關鍵字);
原因:處理靜態成員變數;
使用方法:直接用 static 修飾 普通成員函式(類內宣告時),不需要 static 修飾(類外定義時);
總結:
案例分析:
1 /** 2 * 統計某班選修課考試的平均成績 3 */ 4 5 #include <iostream> 6 #include <string> 7 8 using namespace std; 9 10 class Student 11 { 12 private: 13 string name; // 姓名 14 string gender; // 性別 15 float score; // 分數 16 string subject; // 課程 17 static int total_counts; // 總人數 18 static float chinese_scores; // 語文分數 19 static int chinese_counts; // 語文課人數 20 static float math_scores; // 數學分數 21 static int math_counts; // 數學課人數 22 public: 23 Student(string name, string gender, float score, string subject); 24 ~Student(); 25 static float aveScores(string subject); 26 void printStudentInfo(); 27 void printAveScores(); 28 }; 29 int Student::total_counts = 0; 30 float Student::chinese_scores = 0; 31 int Student::chinese_counts = 0; 32 float Student::math_scores = 0; 33 int Student::math_counts = 0; 34 35 Student::Student(string name, string gender, float score, string subject) 36 { 37 this->name = name; 38 this->gender = gender; 39 this->score = score; 40 this->subject = subject; 41 42 if(subject == "chinese" || subject == "語文") 43 { 44 chinese_scores += score; 45 chinese_counts++; 46 } 47 else if(subject == "math" || subject == "數學") 48 { 49 math_scores += score; 50 math_counts++; 51 } 52 else 53 { 54 cout << "this is no the subect:" << subject << endl; 55 } 56 total_counts += (chinese_counts + math_counts); 57 } 58 59 Student::~Student() 60 { 61 total_counts--; 62 chinese_counts--; 63 math_counts--; 64 } 65 66 float Student::aveScores(string subject) 67 { 68 float ave_score = 0; 69 70 if(chinese_counts > 0 && subject == "chinese" || subject == "語文") 71 { 72 ave_score = (chinese_scores / chinese_counts); 73 //cout << subject << "\t" << chinese_counts << "\t" << chinese_scores << endl; 74 } 75 else if(math_counts > 0 && subject == "math" || subject == "數學") 76 { 77 ave_score = (math_scores / math_counts); 78 //cout << subject << "\t" <<math_counts << "\t" << math_scores << endl; 79 } 80 81 return ave_score; 82 } 83 84 void Student::printStudentInfo() 85 { 86 cout << name << "\t" << gender << "\t" << score << "\t" << subject << endl; 87 } 88 89 void Student::printAveScores() 90 { 91 cout <<subject << " average score: " << aveScores(subject) << endl; 92 } 93 94 int main(int argc, char const *argv[]) 95 { 96 const int SIZE = 5; 97 Student stu[SIZE] = 98 { 99 Student("10001", "male", 92, "語文"), 100 Student("10002", "male", 91, "數學"), 101 Student("10003", "male", 91, "數學"), 102 Student("10004", "male", 93, "語文"), 103 Student("10005", "male", 92, "語文"), 104 }; 105 106 for(int i = 0; i < SIZE; i++) 107 { 108 stu[i].printStudentInfo(); 109 } 110 111 stu[0].printAveScores(); 112 stu[1].printAveScores(); 113 114 cout << "語文" << " average score: " << Student::aveScores("語文") << endl; 115 cout << "數學" << " average score: " << Student::aveScores("數學") << endl; 116 117 return 0; 118 }靜態成員的案例分析
&n