Will Rogers也一定会说:“没有自由变量这种东西。”
——《C现代》
C语言是面向过程的,关注的是过程,分析出求解问题的步骤,通过函数调用逐步解决问题。
C++是基于面向对象的,关注的是对象,将一件事情拆分成不同的对象,靠对象之间的交互完成。
因为是初学者,博主无法把这些编程思想讲得太深入,大家有兴趣可以看下面那篇文章。
面向过程 VS 面向对象
大家都知道C++最开始叫C with Classes,意思是带类的C。所以C++最开始引入了类,你知道写C++的大佬一开始是怎么引入类的吗?他是用struct来引入的。
我们C语言以前要定义一个学生是用结构体,然后定义姓名、性别、年龄等成员变量,这种方式只能定义成员。而C++在兼容C struct的用法的基础上,同时把struct升级成了类。这个意义有两方面:
这个时候,可能有小可爱要问了,为什么成员变量声明在函数后面,C++编译器编译时不是向上找的吗?
其实成员变量声明在类的任意位置都可以,这里不受影响。因为类有一个概念:类是一个整体。比如说strcpy中有一个_name,它会在类的这个整体里去搜索,所以成员变量声明在哪不受影响。
虽然C++可以用struct做类,但是C++还是更喜欢用一个新的关键字:class
。
class className
{
// 类体:由成员函数和成员变量组成
}; // 一定要注意后面的分号
class为定义类的关键字,ClassName为类的名字,{}中为类的主体,注意类定义结束时后面分号。
类中的元素称为类的成员:类中的数据称为类的属性或者成员变量; 类中的函数称为类的方法或者成员函数。
这个时候Student就是类型,s2我们以前喜欢叫它变量,但是从现在开始,我们就不喜欢叫变量了,我们喜欢叫它对象。
那用struct和class定义类有没有什么区别呢?ok,有的。
C++中有一个概念叫封装,那它就提出一个东西叫访问限定符,它的意思是我类里面的东西不是全部都可以给外部用户访问或者使用的。
如果刚刚我们用class写的类没有加public,会编译不通过。
诶,我们看见报错说在两个函数是私有的,可是我没有说他们是私有的呀。
可见class的默认访问权限为private,struct默认为public(因为struct要兼容C)。
从字面意思可见,public就是可以直接在类外面进行访问,private就是在类外面访问不了。
那现在大家猜一猜struct能不能加访问限定符?ok是可以的。
我们一般定义访问限定符不是说只能定义一个,一般喜欢定义多个访问限定符。比如说在学生这个类里我不想有人来改学生的姓名、性别、年龄,我们就把它们设置为私有的,而只把函数提供给外部用户。
访问限定符除了public、private,还有一个protected。现阶段我们认为protected和private是一样的,protected和private修饰的成员在类外不能直接被访问。 我们以后学到继承的时候,它们的意义就不一样。
总结:
};
为止。**面向对象的三大特性:封装、继承、多态。**继承和多态是我们以后要学的,我们今天先来讲一讲封装。
封装的意义是什么呢,这就要和C语言进行对比。这里博主分别用C语言和C++来定义一个简单的栈来讲解。
封装:将数据和操作数据的方法进行有机结合,隐藏对象的属性和实现细节,仅对外公开接口来和对象进行交互。
封装本质上是一种管理:我们如何管理兵马俑呢?比如如果什么都不管,兵马俑就被随意破坏了。那么我们首先建了一座房子把兵马俑给封装起来。但是我们目的全封装起来,不让别人看。所以我们开放了售票通道,可以买票突破封装在合理的监管机制下进去参观。类也是一样,我们使用类数据和方法都封装到一下。不想给别人看到的,我们使用protected/private把成员封装起来。开放一些共有的成员函数对成员合理的访问。所以封装本质是一种管理。
类定义了一个新的作用域,类的所有成员都在类的作用域中。在类体外定义成员,需要使用 :: 作用域解析符指明成员属于哪个类域。这个和命名空间域类似。
// 两个类域
class Stack
{
public:
void Push(int x)
{}
};
class Queue
{
public:
void Push(int x)
{}
};
那么大家想想这两个Push能同时存在吗?ok,可以,因为它俩不在同一个作用域,分别属于不同的两个类域。
如果我们写项目,代码多一些的时候,我们要声明和定义分离。像C语言我们在.h文件里可以看见数据的结构、我定义了哪些函数、我定义了哪些宏等等。我们想看具体实现时,我们才会去看.c文件。
那C++是怎么分离的呢?
C++主要是定义类,那我们怎么声明呢?如果按照以前的方法会报一堆错误:
有的小可爱会说,这里的报错是因为我们把成员变量设置为了私有。不是这样的,真正的原因是我们在定义的时候没有指明作用域。
此时尽管成员变量是私有的,但是一旦指明类域,那么编译器就知道定义的函数是属于类这一个整体,在类里面是不会限定私有,公有的,类外面才会限定。
当然短小函数你也可以直接在类里面定义:
在这个地方又有一个小的概念:在类里面定义的函数默认是内联函数。 所以一般情况,短小函数可以直接在类里面定义,长一点的函数声明和变量分离。
那么再问大家另外一个问题,我们写的类里面的成员变量是声明还是定义?
ok,大家想清楚变量的定义和声明的区别是什么?
有没有开辟k空间,没有开辟空间就是声明,开辟了空间就是定义。
所以类成员变量仅仅只是声明,只有建立对象(以前所说的变量)时才被定义。
用类创建对象的过程,称为类的实例化
大家想想我们刚才写的栈类是多大呢?类中既可以有成员变量,又可以有成员函数,那么一个类的对象中包含了什么?如何计算一个类的大小?我们先来看看结果:
通过结果来看,类对象的存储方式是只保存成员变量,成员函数存放在公共的代码段。
为什么这样设计?
因为每个对象中成员变量是不同的,但是调用同一份函数,如果存储函数,当一个类创建多
个对象时,每个对象中都会保存一份代码,相同代码保存多次,浪费空间。
我们再通过对下面的不同对象分别获取大小来分析看下:
结论:一个类的大小,实际就是该类中”成员变量”之和,当然也要进行内存对齐,注意空类的大小,空类比较特殊,编译器给了空类一个字节(不是为了存储有效数据,而是占位,表示对象存在过)来唯一标识这个类。
这个其实学习结构体时我们就讲过了,复习一下:
我们先来定义一个日期类Date
我们知道这两个对象调用的Init、Print是同一个函数,那么问题来了:那编译器怎么知道第一个就是2022-5-14,第二个就是2022-5-12呢?编译器是如何知道处理的应该是哪个对象的呢?
C++中通过引入this指针解决该问题,即:C++编译器给每个“非静态的成员函数“增加了一个隐藏的指针参数,让该指针指向当前对象(函数运行时调用该函数的对象),在函数体中所有成员变量的操作,都是通过该指针去访问。只不过所有的操作对用户是透明的,即用户不需要来传递,编译器自动完成。
那形参和实参我能不能自己显示地写出来呢?这是不行的,因为这是编译器隐式处理的,你不能抢了编译器的饭碗。
this指针的特性:
this指针的类型:类的类型* const
如上所示的this指针应该是这样Date* const this
,const加在*之后表示指针本身不能被修改,指针指向的对象可以被修改。
this指针本质上其实是一个成员函数的形参,是对象调用成员函数时,将对象地址作为实参传递给this形参。所以对象中不存储this指针。
this指针是成员函数第一个隐含的指针形参,一般情况由编译器通过ecx寄存器自动传递,不需要用户传递。
大家再来看看两道面试题: