C++的一些小总结 类 静态成员变量/函数 this指针_c++ class 静态指针函数-程序员宅基地

技术标签: 编程语言C++  c++  语言  编译器  string  存储  class  



C++总结之一



比较是学习的一个很有效的方法。c 被认为是最贴近机器语言和执行效率最高的语言,语法规则也不复杂。关于C的知识就懒得写了。


C++是在c的基础上扩展了很多。现在就比较一下:


1。C++有了类的概念,有了面向对象的概念。语言本身是人设计的,为了实现这些变化,操作系统本身是做了很多的改动的。开发语言离开操作系统就变得没有意义。


     正如大家知道,win32系统有大量的C++运行库,每个在Win32系统上跑的程序都会调用到这些C++运行库。所以C++这个语言个人认为对于在win32上的开发是非常重要的。退一步说,即便不在win32上作开发,C++这个语言的设计本身就足以提供大量的设计思想和模范。这些都是很宝贵的。


2。类的概念貌似非常容易,无非是一个带属性和操作的结构体,使用时,声明一个变量为该类,就可以了。但实际这里面有些问题必须彻底搞清楚,才可以写出让自己放心的程序。自己放心了,胆气就足了。因为C++在C之后开发出来的,C的很多毛病在C++上得到了一定的纠正。例如C程序员经常会犯的毛病,变量不做初始化就使用。


    C++为了这个毛病就设立了一个构造函数的东西出来。我认为从这个角度来理解一些东西,才更有体会,而不再是象个机器人一样拼命跟着流行跑,跑得气喘吁吁,却依然不会设计程序。


我决定换一个写作的角度,打破这种逐一介绍各个知识点的方法。从一个点出发,将与之关联的各个知识点串联起来,这或许更符合我们的思维方式。


1。1现在就由面向对象开始吧!


好了,我们开始面向对象了,试想会遇到什么问题?例如我定义了A类,并声明a1,a2是A的实例。当操作系统加载我的程序时,a1,a2的程序代码是一样的,也就是它们都在代码区,但是a1,a2有各自不同的数据,这些数据和代码是分开的,当执行代码时,我如何知道去取a1和a2的各自的变量呢?这就牵扯到了我们第一个知识点,this指针。this指针就是类的实例起始地址——实例的变量存储区的起始地址,和代码区是分开的。有了this指针我就可以访问我的实例的变量,并传递给代码区区执行。值得注意的是,this指针本身并不是实例的一部份,所以不会影响sizeof(实例)的大小。


看下面这个有意思的例子:


虽然编译器会给这两个函数传递this指针,但是它们并没有通过this指针来访问类的成员变量,因此call 2和call 3两行代码可以正确调用;而对于成员函数Test4()要访问类的成员变量,因此要使用this指针,这个时候发现this指针的值为NULL,就会造成程序的崩溃。   
你在调用一个类的成员函数时,编译器会把该实例的this指针当作参数传递给该函数,如果出现访问成员变量,就会根据该指针去寻址。如果不幸该指针为空——娃哈哈,程序崩溃。在VC6之上的开发环境中,你可以把一小部分代码编译成汇编语言,如果你这么做,你会发现,this指针跟一般的函数参数的区别:一般的函数参数是直接压入栈中(push 0Dh),而this指针却被放到了ecx寄存器中。
很显然,this指针的作用域只在类内部,不可能由它去找到其他的变量。


 


谈到类的数据,牵扯到另一个东西——静态成员变量。静态就是在程序编译时就分配存储空间,而不是等到运行时才知道,也就是程序加载之后,它的存储空间就已经分配了。为什么要做这种设计呢?因为我们有时候希望即便类的实例销毁了,但我却希望保存它的某些数据,那么这个时候,静态成员变量就是一个很好的选择。


 对于被所有类共享的数据可声明成静态变量,它在程序运行前就已经存在了(所以一定要初始化),它被放到哪里去了呢?静态存储区,按照出现次序被初始化。 它被所有该类的对象共享,静态变量可以被改变。


tips:


1。静态变量可以被继承,如果子类想屏蔽,也可以定义一个一样的静态变量。
2。初始化: int ClassName :: varname=90;  看到了,前面没有static。它只能在类体外进行。(写错了,把"只能"写成"不能",现在改正,在体内初始化,肯定编译报错。)


    简单的常量可以在类内进行定义,不过我认为还是分开来好:象函数一样,体内声明,体外实现(初始化)。之所以这样做的原因很简单,静态变量不依赖任何一个具体的实例而存在,所以即便没有初始化,它也应该早早的存在,而不能等到实例初始化的时候才赋值。


   在这里有个例外就是私有的静态变量,既然它们属于某个具体实例私有数据,很显然在类外是不可访问的.于是在类外的赋值语句将不起作用,只起到一个"实现"的作用.
3。静态变量调用的时候可以通过对象调用,也可以通过类直接调用。


     class MyTestClass
{
public:
    MyTestClass() : m_ciInt(
1), m_csStr("MyStr") // const成员变量,在ctor参数列表中初始化
    {}
public:
   
const int m_ciInt;
   
const String m_csStr;
   
static int m_siInt;
   
static String m_ssStr;
   
const static int m_csiInt;
   
const static String m_cssStr;
};
int MyTestClass::m_siInt=1;// static成员变量,在外部定义
String MyTestClass::m_ssStr ="MyStr";// static成员变量,在外部定义
constint MyTestClass::m_csiInt=1// const static/static const成员变量,在外部定义
const String MyTestClass::m_cssStr="MyStr";// const static/static const成员变量,在外部定义


注意看:m_ciInt,m_csStr和m_csiInt,m_cssStr是不一样的, m_ciInt(1), m_csStr("MyStr")  是初始化,对于基类和const 变量必须这样做。
m_csiInt = 1,m_cssStr = "MyStr" 是赋值。构造函数是为了构造对象而定义的,而静态变量是 一个类的变量,而不是类的实例的变量,所以自然不能在用于构造对象的构造函数中初始化静态变量。


 1.2 类的静态函数


接下来又是一个稍微有点搞的东东——静态函数。不过一旦理解了,就会觉得很简单。
哎!写的真累,不过把静态变量和静态函数弄清楚,会让你的C++打下坚实的基础,稍安勿躁,写完这个就去休息。。。


静态函数和静态变量很相似,也是属于类而非属于类实例的,先把规则写一下,再来解释为什么是这样。



  • 静态函数没有this指针
    静态函数只能访问类中的静态成员变量
    静态函数不需要类的实列就可以调用


一般函数


CCLP *pclp = new CCLP;
pclp->YourProceudre();


静态函数


CCLP::staticprocedure();



  • 类的成员函数想作为回调函数来使用,一般情况下只能将它定义为静态成员才行。
    比方说,光靠类中”其他“函数,不能将类中的某个“其他”函数当作线程的入口地址。

  • 静态函数经过链接器后名字改编不带class名,不带this指针,类函数改编后和class名有关。

很显然调用静态函数效率要高,因为不需要找类实例的数据就可以执行(所以自然不需要标记类实例地址的this指针了),由于没有隐含的this指针,自然无法访问类的其他变量,而只能访问静态变量了。



最后,如果要作线程函数的入口地址,一定要声明为静态的函数,静态函数的地址是加载即确认,OS可以直接找到,建立新线程。而非静态函数却是动态加载,如果作为回调函数和线程函数,就只能是静态函数.





(本文完)


版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/berry666/article/details/7684758

智能推荐

解决win10/win8/8.1 64位操作系统MT65xx preloader线刷驱动无法安装_mt65驱动-程序员宅基地

文章浏览阅读1.3w次。转载自 http://www.miui.com/thread-2003672-1-1.html 当手机在刷错包或者误修改删除系统文件后会出现无法开机或者是移动定制(联通合约机)版想刷标准版,这时就会用到线刷,首先就是安装线刷驱动。 在XP和win7上线刷是比较方便的,用那个驱动自动安装版,直接就可以安装好,完成线刷。不过现在也有好多机友换成了win8/8.1系统,再使用这个_mt65驱动

SonarQube简介及客户端集成_sonar的客户端区别-程序员宅基地

文章浏览阅读1k次。SonarQube是一个代码质量管理平台,可以扫描监测代码并给出质量评价及修改建议,通过插件机制支持25+中开发语言,可以很容易与gradle\maven\jenkins等工具进行集成,是非常流行的代码质量管控平台。通CheckStyle、findbugs等工具定位不同,SonarQube定位于平台,有完善的管理机制及强大的管理页面,并通过插件支持checkstyle及findbugs等既有的流..._sonar的客户端区别

元学习系列(六):神经图灵机详细分析_神经图灵机方法改进-程序员宅基地

文章浏览阅读3.4k次,点赞2次,收藏27次。神经图灵机是LSTM、GRU的改进版本,本质上依然包含一个外部记忆结构、可对记忆进行读写操作,主要针对读写操作进行了改进,或者说提出了一种新的读写操作思路。神经图灵机之所以叫这个名字是因为它通过深度学习模型模拟了图灵机,但是我觉得如果先去介绍图灵机的概念,就会搞得很混乱,所以这里主要从神经图灵机改进了LSTM的哪些方面入手进行讲解,同时,由于模型的结构比较复杂,为了让思路更清晰,这次也会分开几..._神经图灵机方法改进

【机器学习】机器学习模型迭代方法(Python)-程序员宅基地

文章浏览阅读2.8k次。一、模型迭代方法机器学习模型在实际应用的场景,通常要根据新增的数据下进行模型的迭代,常见的模型迭代方法有以下几种:1、全量数据重新训练一个模型,直接合并历史训练数据与新增的数据,模型直接离线学习全量数据,学习得到一个全新的模型。优缺点:这也是实际最为常见的模型迭代方式,通常模型效果也是最好的,但这样模型迭代比较耗时,资源耗费比较多,实时性较差,特别是在大数据场景更为困难;2、模型融合的方法,将旧模..._模型迭代

base64图片打成Zip包上传,以及服务端解压的简单实现_base64可以装换zip吗-程序员宅基地

文章浏览阅读2.3k次。1、前言上传图片一般采用异步上传的方式,但是异步上传带来不好的地方,就如果图片有改变或者删除,图片服务器端就会造成浪费。所以有时候就会和参数同步提交。笔者喜欢base64图片一起上传,但是图片过多时就会出现数据丢失等异常。因为tomcat的post请求默认是2M的长度限制。2、解决办法有两种:① 修改tomcat的servel.xml的配置文件,设置 maxPostSize=..._base64可以装换zip吗

Opencv自然场景文本识别系统(源码&教程)_opencv自然场景实时识别文字-程序员宅基地

文章浏览阅读1k次,点赞17次,收藏22次。Opencv自然场景文本识别系统(源码&教程)_opencv自然场景实时识别文字

随便推点

ESXi 快速复制虚拟机脚本_exsi6.7快速克隆centos-程序员宅基地

文章浏览阅读1.3k次。拷贝虚拟机文件时间比较长,因为虚拟机 flat 文件很大,所以要等。脚本完成后,以复制虚拟机文件夹。将以下脚本内容写入文件。_exsi6.7快速克隆centos

好友推荐—基于关系的java和spark代码实现_本关任务:使用 spark core 知识完成 " 好友推荐 " 的程序。-程序员宅基地

文章浏览阅读2k次。本文主要实现基于二度好友的推荐。数学公式参考于:http://blog.csdn.net/qq_14950717/article/details/52197565测试数据为自己随手画的关系图把图片整理成文本信息如下:a b c d e f yb c a f gc a b dd c a e h q re f h d af e a b gg h f bh e g i di j m n ..._本关任务:使用 spark core 知识完成 " 好友推荐 " 的程序。

南京大学-高级程序设计复习总结_南京大学高级程序设计-程序员宅基地

文章浏览阅读367次。南京大学高级程序设计期末复习总结,c++面向对象编程_南京大学高级程序设计

4.朴素贝叶斯分类器实现-matlab_朴素贝叶斯 matlab训练和测试输出-程序员宅基地

文章浏览阅读3.1k次,点赞2次,收藏12次。实现朴素贝叶斯分类器,并且根据李航《统计机器学习》第四章提供的数据训练与测试,结果与书中一致分别实现了朴素贝叶斯以及带有laplace平滑的朴素贝叶斯%书中例题实现朴素贝叶斯%特征1的取值集合A1=[1;2;3];%特征2的取值集合A2=[4;5;6];%S M LAValues={A1;A2};%Y的取值集合YValue=[-1;1];%数据集和T=[ 1,4,-1;..._朴素贝叶斯 matlab训练和测试输出

Markdown 文本换行_markdowntext 换行-程序员宅基地

文章浏览阅读1.6k次。Markdown 文本换行_markdowntext 换行

错误:0xC0000022 在运行 Microsoft Windows 非核心版本的计算机上,运行”slui.exe 0x2a 0xC0000022″以显示错误文本_错误: 0xc0000022 在运行 microsoft windows 非核心版本的计算机上,运行-程序员宅基地

文章浏览阅读6.7w次,点赞2次,收藏37次。win10 2016长期服务版激活错误解决方法:打开“注册表编辑器”;(Windows + R然后输入Regedit)修改SkipRearm的值为1:(在HKEY_LOCAL_MACHINE–》SOFTWARE–》Microsoft–》Windows NT–》CurrentVersion–》SoftwareProtectionPlatform里面,将SkipRearm的值修改为1)重..._错误: 0xc0000022 在运行 microsoft windows 非核心版本的计算机上,运行“slui.ex