• 学习交流加(可免费帮忙下载CSDN资源):
  • 个人微信: liu1126137994
  • 学习交流资源分享qq群1(已满): 962535112
  • 学习交流资源分享qq群2(已满): 780902027
  • 学习交流资源分享qq群3:893215882

在上一篇文章中我们讲了C++中类的静态成员变量,用类的静态成员变量实现了统计程序运行期间的某个类的对象的数目(不清楚的可以点击连接查看上一篇文章 C++中类的静态成员变量)。

我们回顾一下客户的需求:

  • 统计在程序运行期间某个类的对象的数目
  • 保证程序的安全性(不能使用全局变量)
  • 随时可以获取当前对象的数目(完成了么?)

看当时的代码:

#include <stdio.h>class Test
{private:static int cCount;
public:Test(){cCount++;}~Test(){--cCount;}int getCount(){return cCount;}
};int Test::cCount = 0;Test gTest;int main()
{Test t1;Test t2;printf("count = %d\n", gTest.getCount());printf("count = %d\n", t1.getCount());printf("count = %d\n", t2.getCount());Test* pt = new Test();printf("count = %d\n", pt->getCount());delete pt;printf("count = %d\n", gTest.getCount());return 0;
}

很明显,我们获取对象的数目是需要调用getCount这个函数,每次调用都需要某一个对象来调用,比如:gTest.getCount(),那么问题来了,如果整个程序的对象的个数为0呢?那么岂不是没法调用getCount这个函数,也就没法知道对象的个数(别人是不知道你的对象的个数为0的)?

可不可以这样呢?我把静态成员变量cCount变为public,然后直接让Test(作用域调用)这个类来调用?先上一波代码看看:

#include <stdio.h>class Test
{public:static int cCount;
public:Test(){cCount++;}~Test(){--cCount;}int getCount(){return cCount;}
};int Test::cCount = 0;int main()
{printf("count = %d\n", Test::cCount);//Test::cCount = 1000;//printf("count = %d\n", Test::cCount);return 0;
}

此代码与上面的代码的不同的是cCount变量变为public便于Test(作用域访问)类调用,没有任何的对象了,那么编译运行,输出结果应该是:
count = 0

说明这样做,符合了我们的需求了。但是,不要高兴的太早了,我们这样真的可以么?加入我把代码中的被注释的那两行加上(下面这两行):

     Test::cCount = 1000;printf("count = %d\n", Test::cCount);

再次运行会发生什么呢?我们编译运行,结果输出为:

 count = 0count = 1000

那么多问题来了,客户现在懵逼了,他会以为对象的个数有1000个,所以,这也是一个大的Bug啊,从本上讲,这样做也是无法满足客户的需求的,而且将cCount变为public,本身就是无法保证静态成员变量的安全性,我们需要什么?

  • 不依赖对象就可以访问静态成员变量
  • 必须保证静态成员变量的安全性
  • 方便快捷得获取静态成员变量的值

那么静态成员函数就要出马了:
静态成员函数的定义:
直接通过static关键字修饰成员函数即可

为了便于理解,我们先上一段代码来理解一下静态成员函数的性质:

#include <stdio.h>class Demo
{private:int i;
public:int getI();static void StaticFunc(const char* s);static void StaticSetI(Demo& d, int v);
};int Demo::getI()
{return i;
}void Demo::StaticFunc(const char* s)
{printf("StaticFunc: %s\n", s);
}void Demo::StaticSetI(Demo& d, int v)
{d.i = v;
}int main()
{Demo::StaticFunc("main Begin...");Demo d;d.StaticSetI(d, 20);printf("d.i = %d\n", d.getI());Demo::StaticSetI(d, 10);printf("d.i = %d\n", d.getI());Demo::StaticFunc("main End...");return 0;
}

以上代码运行的结果为:

StaticFunc: main Begin...
d.i = 20
d.i = 10
StaticFunc: main End...

由代码看到两个静态成员函数:

static void StaticFunc(const char* s);
static void StaticSetI(Demo& d, int v);

类调用静态成员函数:

Demo::StaticFunc("main Begin...");

对象调用静态成员函数:

d.StaticSetI(d, 20);

可以看出静态成员函数的性质大体如下:

  • 静态成员函数是类中特殊的成员函数
  • 静态成员函数属于整个类所有
  • 可以通过类名(作用域访问)直接访问公有静态成员函数
  • 可以通过对象名访问公有静态成员函数

而且通过这段代码:

void Demo::StaticSetI(Demo& d, int v)
{d.i = v;
}

我么可以看出,静态成员函数,没有直接调用变量i,而是通过对象来间接调用变量i,这说明什么呢?说明:静态成员函数不能访问普通成员变量(函数),需通过对象间接访问成员变量(函数)
下面做一个表格来对比一下静态成员函数与普通成员函数的区别:

好了,理解了静态成员函数,我们也应该来解决客户的需求了吧:直接上代码:

#include <stdio.h>class Test
{private:static int cCount;
public:Test(){cCount++;}~Test(){--cCount;}static int GetCount(){return cCount;}
};int Test::cCount = 0;int main()
{printf("count = %d\n", Test::GetCount());Test t1;Test t2;printf("count = %d\n", t1.GetCount());printf("count = %d\n", t2.GetCount());Test* pt = new Test();printf("count = %d\n", pt->GetCount());delete pt;printf("count = %d\n", Test::GetCount());return 0;
}

运行结果为:

count = 0
count = 2
count = 2
count = 3
count = 2

分析: printf("count = %d\n", Test::GetCount());这段代码运行之前,没有创建对象,所以count = 0,

printf("count = %d\n", t1.GetCount());
printf("count = %d\n", t2.GetCount());

这两段代码运行之前已经创建了t1,t2这两个对象,所以都为:count = 2,

printf("count = %d\n", pt->GetCount());

这个运行之前又在堆空间创建了一个对象,并且让pt指针指向它,所以:count = 3

 delete pt;printf("count = %d\n", Test::GetCount());

在这两行先将pt指向的对象删除,所以最后是:count = 2

问题终于解决了!!!

总结:

  • 静态成员函数是类中的特殊的成员函数
  • 静态成员函数没有隐藏的this指针
  • 静态成员函数可以通过类名直接访问
  • 静态成员函数可以通过对象访问
  • 静态成员函数只能直接访问静态成员变量(函数),而不能直接访问普通成员变量(函数)

想获得各种学习资源以及交流学习的加我:
qq:1126137994
微信:liu1126137994
可以共同交流关于嵌入式,操作系统,C++语言,C语言,数据结构等技术问题!

【C++深度剖析教程5】C++中类的静态成员函数相关推荐

  1. 【C++深度剖析教程3】C++中类的静态成员变量

    学习交流加(可免费帮忙下载CSDN资源): 个人微信: liu1126137994 学习交流资源分享qq群1(已满): 962535112 学习交流资源分享qq群2: 780902027 以一个简单的 ...

  2. 【C++深度剖析教程39】实现C++数组类模板

    上一篇文章在那个学习了多参数类模板与特化的分析:点击链接查看上一篇文章:类模板深度剖析 本篇文章学习记录: 数值型模板参数 实现C++数组类模板 1.模板中的数值型参数 模板参数可以是数值型参数.也就 ...

  3. 【C++深度剖析教程7】C++之类中的函数重载

    函数重载的回顾(接上一篇文章): 函数重载的本质为相互独立的不同的函数 C++中通过函数名和函数参数确定函数调用 无法直接通过函数名得到重载函数的入口地址 函数重载必然发生在同一个作用域中 类中的成员 ...

  4. 【C++深度剖析教程38】类模板深度剖析

    加qq1126137994 微信:liu1126137994 一起学习更多技术!!! 1.多参数类模板 类模板可以定义任意多个不同的类型参数 类模板可以被特化: 指定类模板的特定实现 部分类型参数必须 ...

  5. 【C++深度剖析教程25】继承中的构造与析构

    今天来学习C++中继承的构造与析构,有兴趣一起学习的加qq:1126137994 1.问题 如何初始化父类成员?父类构造函数与子类构造函数有什么关系? 子类对象是如何构造的? 子类中可以定义构造函数 ...

  6. 【C++深度剖析教程22】继承的概念和意义

    今天我们来学习C++中的继承的概念和意义. 一.类之间的组合关系 问题:类之间是否存在直接的关联关系? 回答:类之间存在组合的关系,整体与部分的关系. 可以看一下生活中的例子: 下面我们以一个简单的程 ...

  7. 【C++深度剖析教程8】C++的操作符重载的概念

    之前学习了类的函数重载的概念,今天学习操作符重载的概念.在这之前我们先看一个例子: 上面是一个复数的加法,a为复数的实部,b为复数的虚部,在main函数里我想实现复数c1与c2的加法.很显然,正常的+ ...

  8. 【C++深度剖析教程40】使用数值型模板技术计算1+2+3+...+N的值

    上一篇文章学习了数值型模板技术,并利用相关技术,实现了C++的数组类模板.点击文章查看上一篇文章:点击链接查看 本篇文章,继续利用模板技术来解决一个问题. 如果想求1+2+3+-+N的结果,有很多种方 ...

  9. 【C++深度剖析教程37】类模板的概念和意义

    加qq1126137994 微信:liu1126137994 一起学习更多技术!!! 1.类模板 一些类主要用于存储和组织数据元素 类中数据的组织方式和数据元素的具体类型无关 如 数组类,链表类,st ...

最新文章

  1. 二叉树(C++):创建,前中后序遍历(递归+非递归),获取叶子节点个数,获取树的高度
  2. Centos 配置JAVA_HOME
  3. 正则表达式30分钟入门教程-2
  4. elasticsearch5.3安装kibana、x-pack插件、elasticsearch-an
  5. 无人驾驶(基于计算机视觉的高精度地图)
  6. leetcode python3 简单题136. Single Number
  7. paip.提升用户体验-------在C++ Builder 中为Form窗体添加背景图片
  8. 过流媒体取流失败_海康硬盘录像机:监控点取流失败,开始重连.错误代码为iVMS-4200.EXE[302]求大神解决...
  9. html 弹出框显示到最顶层,layer弹出层显示在top顶层的方法
  10. JavaScript基础复习下(51st)
  11. 定义图书类Book,具有属性账号id,铭name.作者author和价格price,在创建图书对象时要求通过构造器进行创建,- -次性将四个属性全部赋值
  12. ikbc c104win键盘失灵以及数字键失灵
  13. 09.CSS3渐变、过渡、转换、动画
  14. Macbook Pro(MBP)上固态硬盘SSD,光驱位装HDD
  15. LoRa 扩频因子和码片
  16. matlab使用linprog()函数解决简单的线性规划问题
  17. YOLO v3算法解析
  18. 解决Win10局域网共享问题:请检查名称的拼写 否则 网络...
  19. 一文了解CPU及芯片硬件技术发展
  20. 微信iPad协议-最新完整版

热门文章

  1. spring学习(32):使用junit4测试
  2. php 支付加密,关于支付时rsa加密解密的函数
  3. stylus之关键字参数(Keyword Arguments)
  4. android imageview 图片切换动画,在Android中以动画方式将ImageView移动到不同的位置...
  5. dbnetlib sqlserver不存在或拒绝访问_SQL Server数据库损坏和修复
  6. React 在body上绑定事件以及阻止事件冒泡
  7. SQLServer数据库,表内存,实例名分析SQL语句
  8. 字体小于12px解决办法
  9. 【c++ primer读书笔记】【第2章】变量和基本类型
  10. 面向对象之多态性(基类引用可以指向子类)