C++ 类的静态成员及静态成员函数

  •   对象与对象之间的成员变量是相互独立的。要想共用数据,则需要使用静态成员和静态方法。
  •   只要在类中声明静态成员变量,即使不定义对象,也可以为静态成员变量分配空间,进而可以使用静态成员变量。(因为静态成员变量在对象创建之前就已经被分配了内存空间)
  •   静态成员变量虽然在类中,但它并不是随对象的建立而分配空间的,也不是随对象的撤销而释放(一般的成员在对象建立时会分配空间,在对象撤销时会释放)。静态成员变量是在程序编译时分配空间,而在程序结束时释放空间。
  •   静态成员的定义和声明要加个关键static。静态成员可以通过双冒号来使用,即<类名>::<静态成员名>。
  •   初始化静态成员变量要在类的外面进行。初始化的格式如下:数据类型  类名::静态成员变量名 = 初值;
  •   不能用参数初始化表,对静态成员变量进行初始化。
  •   既可以通过类名来对静态成员变量进行引用,也可以通过对象名来对静态成员变量进行引用。
  •   普通成员函数和静态成员函数的区别是:普通成员函数在参数传递时编译器会隐藏地传递一个this指针.通过this指针来确定调用类产生的哪个对象;但是静态成员函数没有this指针,不知道应该访问哪个对象中的数据,所以在程序中不可以用静态成员函数访问类中的普通变量.

  下面通过几个例子来总结静态成员变量和静态成员函数的使用规则。

  一、通过类名调用静态成员函数和非静态成员函数

 1 //例子一:通过类名调用静态成员函数和非静态成员函数
 2 class Point{
 3 public:
 4     void init()
 5     {}
 6
 7     static void output()
 8     {}
 9 };
10
11 void main()
12 {
13     Point::init();
14     Point::output();
15 }

  编译出错:错误 1 error C2352: “Point::init”: 非静态成员函数的非法调用

  结论一:不能通过类名来调用类的非静态成员函数

  二、通过类的对象调用静态成员函数和非静态成员函数

 1 //例子二:通过类的对象调用静态成员函数和非静态成员函数
 2 class Point{
 3 public:
 4     void init()
 5     {
 6     }
 7
 8     static void output()
 9     {}
10 };
11
12 void main()
13 {
14     Point pt;
15     pt.init();
16     pt.output();
17 }

  编译通过。

  结论二:类的对象可以使用静态成员函数和非静态成员函数。

  三、在类的静态成员函数中使用类的非静态成员

 1 //例子三:在类的静态成员函数中使用类的非静态成员
 2 #include <iostream>
 3 using namespace std;
 4
 5 class Point{
 6 public:
 7     void init()
 8     {
 9     }
10     static void output()
11     {
12         cout << "m_x=" << m_x << endl;
13     }
14 private:
15     int m_x;
16 };
17 void main()
18 {
19     Point pt;
20     pt.output();
21 }

  编译出错:IntelliSense: 非静态成员引用必须与特定对象相对

  因为静态成员函数属于整个类,在类实例化对象之前就已经分配空间了,而类的非静态成员必须在类实例化对象后才有内存空间,所以这个调用就会出错,就好比没有声明一个变量却提前使用它一样。

  结论三:静态成员函数中不能引用非静态成员。

  四、在类的非静态成员函数中使用类的静态成员

 1 //例子四:在类的非静态成员函数中使用类的静态成员
 2 #include <iostream>
 3 using namespace std;
 4
 5 class Point{
 6 public:
 7     void init()
 8     {
 9         output();
10     }
11     static void output()
12     {
13     }
14 private:
15     int m_x;
16 };
17 void main()
18 {
19     Point pt;
20     pt.init();
21 }

  编译通过。

  结论四:类的非静态成员可以调用静态成员函数,但反之不能。

  五、使用类的静态成员变量

 1 //例子五:使用类的静态成员变量
 2 #include <iostream>
 3 using namespace std;
 4
 5 class Point{
 6 public:
 7     Point()
 8     {
 9         m_nPointCount++;
10     }
11     ~Point()
12     {
13         m_nPointCount++;
14     }
15     static void output()
16     {
17         cout << "m_nPointCount=" << m_nPointCount << endl;
18     }
19 private:
20     static  int m_nPointCount;
21 };
22
23 void main()
24 {
25     Point pt;
26     pt.output();
27 }

  

  链接出错:error LNK2001: 无法解析的外部符号 "private: static int Point::m_nPointCount" ([email protected]@@0HA)

  这是因为类的成员变量在使用前必须先初始化。

  改成如下代码即可:

 1 #include <iostream>
 2 using namespace std;
 3
 4 class Point{
 5 public:
 6     Point()
 7     {
 8         m_nPointCount++;
 9     }
10     ~Point()
11     {
12         m_nPointCount++;
13     }
14     static void output()
15     {
16         cout << "m_nPointCount=" << m_nPointCount << endl;
17     }
18 private:
19     static  int m_nPointCount;
20 };
21
22 //类外初始化静态成员变量时,不用带static关键字
23 int Point::m_nPointCount = 0;
24 void main()
25 {
26     Point pt;
27     pt.output();
28 }

  运行结果:

  

  结论五:类的静态成员变量必须先初始化再使用。

  

  

时间: 2024-10-06 12:16:14

C++ 类的静态成员及静态成员函数的相关文章

静态成员函数为什么不能访问本类中的非静态成员?

和静态数据成员一样,静态成员函数是类的一部分,而不是对象的一部分.如果要在类外调用公用的静态成员函数,要用类名和域运算符"∷".如Box∷volume( );实际上也允许通过对象名调用静态成员函数,如a.volume( );但这并不意味着此函数是属于对象a的,而只是用a的类型而已.静态成员函数的作用是为了能处理静态数据成员.可以说,静态成员函数与非静态成员函数的根本区别是:非静态成员函数有this指针,静态成员函数并不属于某一对象,它与任何对象都无关,静态成员函数没有this指针.由此

类的静态成员变量及函数(二十)

我们正常情况下能通过对象名来访问 public 成员变量,每个对象的成员变量都是专属的,成员变量不能在对象之间共享的.我们现在学了这么长时间的 C++,现在又有了一个新的需求:a> 同级在程序运行期间某个类的对象数目:b> 保证程序的安全性(不能使用全局变量):c> 随时可以获取当前对象的数目. 我们首先想到的是定义一个私有成员变量,然后在构造函数中进行 ++ 操作,在析构函数中进行 -- 操作.我们试着写下程序 #include <stdio.h> class Test {

析构函数 this 静态成员变量 静态成员函数

1. 析构函数 1.    析构函数没有参数,不能被重载,因此一个类只能有一个析构函数.如果用户没有定义,那么编译器会自动生成. 2.  static 局部对象在函数调用结束时并不销毁,因此也不调用析构函数, 只有在程序结束时(如 main 函数结束或调用 exit 函数)才调用 static 局部对象的析构函数. 3) 如果定义了一个全局对象,也只有在程序结束时才会调用该全局对象的析构函数. #include <iostream> #include <cstdlib> using

对【面向对象的类访问和对象访问的区别【this以及类访问】、静态成员的访问区别、类常量等、继承和重写、访问修饰限定符、冒泡排序】的总结

面向对象的总结1.首先面向对象的编程使用的都是使用对象作为处理对象而言[例如继承等情形,继承的本质来说,依旧针对的是对象]但是只有针对类的访问.静态成员以及访问修饰限定符时,针对的是类的概念 2.类内定义时,只有五种情形:类常量[const关键字定义并且使用类来调用]属性[这些属性和方法是针对该类实例的对象来调用]方法[在这种情形下使用$this进行指代作用环境(调用该方法的对象),只有继承中,子类实例的对象会向下传递到静态调用里]静态属性[用来作为实例该类的所有对象之间共享的数据.保存在类中]

C++ Primer 学习笔记33_面向对象编程(4)--虚函数与多态(一):多态、派生类重定义、虚函数的访问、 . 和-&gt;的区别、虚析构函数、object slicing与虚函数

C++ Primer学习笔记33_面向对象编程(4)--虚函数与多态(一):多态.派生类重定义.虚函数的访问. . 和->的区别.虚析构函数.object slicing与虚函数 一.多态 多态可以简单地概括为"一个接口,多种方法",前面讲过的重载就是一种简单的多态,一个函数名(调用接口)对应着几个不同的函数原型(方法). 更通俗的说,多态行是指同一个操作作用于不同的对象就会产生不同的响应.或者说,多态性是指发出同样的消息被不同类型的对象接收时有可能导致完全不同的行为. 多态行分

在C++的类中,普通成员函数不能作为pthread_create的线程函数,如果要作为pthread_create中的线程函数,必须是static

在C++的类中,普通成员函数不能作为pthread_create的线程函数,如果要作为pthread_create中的线程函数,必须是static ! 在C语言中,我们使用pthread_create创建线程,线程函数是一个全局函数,所以在C++中,创建线程时,也应该使用一个全局函数.static定义的类的成员函数就是一个全局函数. 更多 参考  http://blog.csdn.net/ksn13/article/details/40538083 #include <pthread.h> #

C++之中this指针与类的六个默认函数小结

我们先来看看this指针.之前看过一篇关于this指针的文章,觉得写的很好,今天决定自己来写一写,顺便总结一下C++里面关于类的一些内容. 什么是this指针呢?简单的说它是一个指向类的实例的指针,就好像当我们在进入一个房子之后,可以看见房子里的桌子,椅子.地板等, 但是看不到房子的全貌.对于一个类的实例来说,你可以看到它的成员函数.成员变量,但是实例本身呢?this是一个指针,它时时刻刻指向这个实例. 来看看this指针的特性: 1)this指针的类型是一个类类型 * const, 这表示什么

浅析在类模版中构建成员函数时,使用memcpy产生的副作用

一般情况下我们在对类模版中的成员函数进行构建时会经常对一些数据进行复制拷贝,而通常情况下我们都不提倡用memcpy进行拷贝,因为在类模版中所传进来的类型可以是内置类型也可以是非内置类型,除非你在成员函数中使用memcpy前进行类型萃取,否则它所带来的副作用的后果也是很可怕的.memcpy在对内置类型可正常拷贝,而在对非内置类型拷贝时会出现浅拷贝的现象. 下面我们可以通过一个简单的顺序表程序来分析memcpy对非内置类型所产生的副作用: #include<iostream> #include&l

(C++)浅谈多态基类析构函数声明为虚函数

主要内容: 1.C++类继承中的构造函数和析构函数 2.C++多态性中的静态绑定和动态绑定 3.C++多态性中析构函数声明为虚函数 1.C++类继承中的构造函数和析构函数 在C++的类继承中, 建立对象时,首先调用基类的构造函数,然后在调用下一个派生类的构造函数,依次类推: 析构对象时,其顺序正好与构造相反: 具体参考文章:http://www.cnblogs.com/AndyJee/p/4575385.html 2.C++多态性中的静态绑定和动态绑定 对象的静态类型:对象在声明是采用的类型,在