C++中静态成员函数与静态成员变量(static )
作者:lqh 发布时间:2021-11-08 07:31:38
C++中静态成员函数与静态成员变量(static )
这篇介绍了静态成员函数与静态成员变量,是我的读书笔记,我希望它够简短但又比较全面,起到复习的作用。如果有一些C++知识记不清楚了,它可以帮你很快回忆起来。
复习C语言的static关键字
(1)加在局部变量的前面使之成为静态局部变量,作用域还是在函数内部,可是生存周期延长了。
(2)加在全局变量的前面限定该变量作用域为文件作用域,就是说即使其他文件使用了extern扩展作用域也不行。这在C语言的多人项目中非常有用,避免了变量的重名。然而在C++中这一功能已经被命名空间取代,但是为了保持和C语言的兼容,static还是有这样的功能。
(3)加在函数定义或声明的前面,限定函数作用域到文件作用域,也是为了避免多个文件中有重名函数。
当static关键字出现在类中
当static出现在类的定义中便出现了静态成员变量和静态成员函数。静态成员是属于类的,而不是属于某个对象的。即便没有任何一个实例,类的静态成员变量也已经存在了,而且还可能通过“类名::成员名”进行访问。类的静态成员函数也可以用相同的方式调用,在类产生实例之前就调用成员方法,典型应用是实现单例模式。
(1)静态成员变量
静态成员变量本质上是全局变量,但是将和某些类关系紧密的全局变量写到类里面,形式上成为一个整体,更容易理解和维护。所以尽量使用静态成员变量吧,减少全局变量的使用。普通成员变量每个对象都有各自的一份,但是静态成员变量一共只有一份,被所有的本类对象共享。如果使用sizeof运算符计算对象的大小,得到的结果是不包含静态成员变量在内的。
静态成员同样受到private,public等的限制。
静态成员变量的一个典型应用就是用来计数生成的实例的个数。大体思路是设置一个名为num的静态成员变量并初始化为0,在构造函数中++num,析构函数中--num。这样num的值就是当前实例的个数。实际上这也带来了一个隐蔽的bug。看下面的代码:
class CNum {
public:
static int num;
~CNum() { --num; }
CNum() { ++num; }
};
int CNum::num = 0;
void fun(CNum n){ }
int main() {
CNum n;
fun(n);
fun(n);
cout << CNum::num << endl;
return 0;
}
结果:-1
num尽然成了一个负数,难道析构函数比构造函数多调用了一次?实际上不是的。当执行 fun(n); 语句时调用了复制构造函数,这个函数因为我们没有给出实现,所以是用的编译器默认提供的版本,在这个构造函数中并没有++num这条语句,因此少计数了两次(两次调用fun(n))。
解决的方法就是一定要提供自己写的复制构造函数并在函数体中加入 ++num;
(2)静态成员函数
静态成员函数内部不能调用非静态成员函数,原因是,非静态成员函数需要传入一个this指针,这让静态成员函数很为难,它并不知道与之相关的信息,也就无法提供this指针。
静态成员变量的初始化
上面代码中的第8行 int CNum::num = 0; 是静态成员变量的初始化。这可以视为是静态变量的定义(定义的同时初始化,即便不初始化也需要这个定义),而把类内的 static int num; 视为一个声明,这样的理解可以突出这样一个事实:静态成员变量本质上是全局变量。注意在类外定义时加上“类名::”。
对于常量成员变量,我们知道初始化时一定要使用初始化列表,那么当一个变量既是常量又是静态成员时(同时被const和static修饰)要怎么样初始化呢?是像一般的静态成员变量一样在类外定义并初始化,还是像一般的常量成员变量一样使用初始化列表呢?答案时前者,即在类外定义并初始化,在类内声明,就像下面那样:
class CNum {
public:
const static int num;
};
const int CNum::num = 0;
int main() {
CNum n;
return 0;
}
实际上,完全可以把const int 视为一种数据类型,它的地位和int一样。这样理解是有好处的,比如从const int到int需要强制类型转换,把他们看成两种类型,这就自然而然。相应的const char 和char 也应该看成两种类型,就好像它们完全没有什么特殊的关系一样。
另外static const int类型和static const char 类型可以在类内直接初始化,就是说都不需要在类外再次定义,像下面这样:
class CNum {
public:
const static int a = 19;
};
int main() { 8 cout << CNum::a << endl;//输出19
return 0;
}
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!
来源:http://www.cnblogs.com/painterQ/p/7009404.html


猜你喜欢
- 目录1)在程序集中添加资源2)在程序集中查找资源这一篇单独拿出来分析这个程序集资源,为的就是不想让大家把程序集资源和exe程序强关联,因为程
- 目录(?)[-]一扩展javalangThread类二实现javalangRunnable接口三Thread和Runnable的区别四线程状
- 当你使用synchronized关键字的时候,是通过互斥器来保障线程安全以及对共享资源的同步访问。线程间也经常需要更进一步的协调
- Java * 分析及理解代理设计模式定义:为其他对象提供一种代理以控制对这个对象的访问。 * 使用java * 机制以巧妙的方式实现了
- Android中操作Excel文件导出报表时主要采用开源库jxl,最早用在java上,但也可用于Android。与之类似的POI,因为依赖库
- 1. 前言Spring最重要的一个概念当属Bean了,我们写的Controller、Service、Dao凡是加了对应注解交给Spring管
- 一、背景项目中肯定会遇到异步调用其他方法的场景,比如有个计算过程,需要计算很多个指标的值,但是每个指标计算的效率快慢不同,如果采用同步执行的
- 本文实例为大家分享了Java实现酒店客房管理系统的具体代码,供大家参考,具体内容如下LoginFrame.javapackage login
- 本文实例为大家分享了C#实现窗体抖动的具体代码,供大家参考,具体内容如下原理:围绕中心点运动一圈方法一:通过线程实现需求:需要using S
- 本文介绍了Maven构建自己的第一个Java后台的方法,分享给大家,具体如下:1.知识后顾关于如何运用Maven构建自己的第一个项目,上期我
- JRebel 介绍IDEA上原生是不支持热部署的,一般更新了 Java 文件后要手动重启 Tomcat 服务器,才能生效,浪费不少生命啊。目
- C语言fchdir()函数:改变当前工作目录头文件:#include <unistd.h>定义函数:int fchdir(int
- 在面向对象设计原则中,要求"要依赖于抽象,不要依赖于具体", 这句话有很多人搞不懂。在这里谈谈我自己的理解。首先看看以下
- 1、添加android support包因为上面的几个类都是在android support包中才提供,我们先添加包。在Eclipse-&g
- 1.概述在实际开发过程中,我们经常需要调用对方提供的接口或测试自己写的接口是否合适。很多项目都会封装规定好本身项目的接口规范,所以大多数需要
- 棋牌类游戏是目前比较火的游戏之一。今天本文就以实例形式实现洗牌游戏。本文实例所采用的算法是:遍历每个位置上的牌,然后与随机位置上的牌交换。运
- 本文实例为大家分享了java斗地主发牌的具体代码,供大家参考,具体内容如下分析这是一个模仿斗地主发牌的例子;按照斗地主的规则,完成洗牌发牌的
- 归并排序原理1.尽可能的一组数据拆分成两个元素相等的子组,并对每一个子组继续拆分,直到拆分后的每个子组的元素个数是1为止。⒉将相邻的两个子组
- SpringMVC AbstractAnnotationConfigDispatcherSerServlet3.0环境中,容器会在类路径中查
- Java自定义注解一般使用场景为:自定义注解+ * 或者AOP,使用自定义注解来自己设计框架,使得代码看起来非常优雅。本文将先从自定义注解的