《类和对象:基础原理全解析(上篇)》

目录

一、浅谈面向过程和面向对象

C 语言是面向过程的语言,而 C++ 是面向对象的语言。并且当前主流的语言大多数都是面向过程的语言,这足以证明面向过程的语言更适合当前时代的发展。

面向过程在解决问题时主要关注解决问题的过程,也就是解决问题的步骤,把每个步骤抽象成一个函数,然后传入对应的数据调用这些函数一步一步解决问题。而面向对象在解决问题时主要关注问题涉及的对象,把每个对象、相关的数据(成员变量)和能进行的操作(成员函数)封装成一个类。接着创建每个类的对象实例用来存储数据和执行相应的操作。

就拿玩游戏闯关打 BOSS 来举例子,如果是面向过程的游戏,那么就需要按照顺序从第一关,第二关,...,第 N 关,BOSS。你需要按照策划设定好的过程一关一关地通过,最后面对 BOSS 。而如果是面向对象的游戏,那么就抽象出来了三个类:玩家、小怪和BOSS,小怪和 BOSS 的设定是如果与玩家的距离在 N 之内或者玩家主动攻击,那么小怪和 BOSS 就会反击。这样游戏就多了很多可能性,比如玩家只通过了一部分关卡就去打 BOSS 或者上来就攻打 BOSS。

通过上述分析可以得出,面向过程的语言关注的是过程,把一个问题分解成多个步骤,调用函数逐步解决。而面向对象是把问题中涉及的对象(包括其属性和操作)抽象出来,通过对象之间的交互来完成。

二、C++ 中的结构体(struct)

C++ 中的 struct 中不仅能包含成员变量还能包含成员函数,因为 C++ 中把 struct 当作类来处理。

1. C++ 中 struct 的使用

下面是一个 C++ 中的学生 struct :

cpp 复制代码
// 头文件
#include <iostream>

// 常量声明
const int SIZE_NAME = 20;
const int SIZE_ID = 11;

// 使用声明
using std::cout;
using std::endl;

// 学生结构声明
struct Student
{
	// 成员变量
	char _name[SIZE_NAME];
	size_t _age;
	char ID[SIZE_ID];

	// 成员函数
	void PrintInfo()
	{
		cout << _name << " " << _age << " " << ID << endl;
	}
};

int main()
{
	// 创建结构体变量
	Student zhangsan = { "张三", 18, "2101240002" };
	// 调用成员函数
	zhangsan.PrintInfo();

	return 0;
}

程序的运行结果如下:

从上面的代码中可以看到在创建结构体变量时,直接使用 Student 而不是 C 语言中的 struct Student,这是因为 C++ 中把结构体当作类处理,而 Student 是一个类名。但是由于 C++ 兼容 C 语言,所以原来 C 语言的 struct Student 语法依旧可以使用。

结构体对象(变量)通过成员运算符(.)来访问成员变量和调用成员函数。

三、C++ 中的类(class)

C++ 中的类由描述类的成员变量和能对类进行操作的成员函数组成。

C++ 中的类的创建类似 struct,只不过是把 struct 改成了 class,把结构体名称改为了类名。

class classname

{

//...

};

下面依旧是学生结构体:

cpp 复制代码
// 头文件
#include <iostream>

// 常量声明
const int SIZE_NAME = 20;
const int SIZE_ID = 11;

// 使用声明
using std::cout;
using std::endl;

// 学生类
class Student
{
	// 成员变量
	char name[SIZE_NAME];
	size_t age;
	char ID[SIZE_ID];

	// 成员函数
	void Print()
	{
		cout << name << " " << age << " " << ID << endl;
	}
};

int main()
{
	Student zhangsan = { "zhangsan", 18, "2101240002" };
	// 调用成员函数
	zhangsan.Print();

	return 0;
}

可以看到上面的代码与 struct 中的代码除了一个是结构体一个是类,其他都没有区别,但是为什么使用类时编译器报错了呢?

上面的问题就涉及类的封装性和六大默认函数了。

四、类的封装性

什么是封装性?就拿王者荣耀来说,在面向对象设计中,玩家是一个类,该类包含了玩家的属性(金币、点券、对局数等信息)和玩家能进行的操作(娱乐、匹配、排位等)。那么玩家的属性就是玩家类的成员变量,玩家能进行的操作就是玩家类的成员函数。玩家可以调用成员函数进行对局,但是玩家不能访问成员变量修改点券,只能通过成员函数(充值)从而间接修改点券。如果可以直接访问玩家属性,那么王者荣耀就无法进行游戏管理和盈利。

类的封装性通过隐藏类的成员变量,展示类的成员函数,让用户通过成员函数来访问成员变量,这样用户的操作就比较规范合理。也可以这样理解,类的设计者可以通过类的封装性,让用户可以看见设计者想让用户看见的,也可以让用户看不见设计者不想让用户看见的。

1. 类成员的权限控制关键字

类成员有 public(公有的)、private(私有的)和 protected(受保护的)。public 成员可以在类外通过类对象进行访问,private 只能在类内进行访问,而在当前的学习阶段只需要把 protected 当作 private 理解就行。protected 在后面的类继承中才会有其他作用。

现在来解释一下为什么上面使用 class 定义的 Student 类不能正常使用,而使用 struct 定义的 Student 却能正常使用。因为如果不显示指明权限,class 默认权限为 private,而 struct 默认权限为 public。

2. 权限控制关键字的使用

下面是使用了权限关键字之后可以正常使用的 Student 类:

虽然现在已经加上了权限关键字,但是上述代码仍存在问题。成员函数 Print() 的权限为 public,所以现在不存在函数调用的问题。而是类对象初始化的问题,这就涉及到六大默认函数中的构造函数了。

五、类的六大默认成员函数介绍

六大默认成员函数函数分别为:构造函数、析构函数、拷贝构造函数、赋值运算符重载、普通对象和 const 对象取地址运算符重载。

为什么称呼为默认成员函数?是因为如果程序员不显式定义这些函数,那么编译器会自动生成对应的默认函数。

主要需要学习前四个默认函数,而最后两个很少自己重新定义。

六、构造函数

构造函数的作用是在创建类对象的同时对其初始化,而不是创建类对象。且构造函数的函数名和类名相同,没有返回值,也不写 void。

且在初始化对象时,无需使用成员运算符(.),直接在后面接上圆括号然后传入对应参数即可(中间用逗号隔开)。如果没有参数,直接创建对象即可,无需使用圆括号(前提是存在默认构造函数)。

构造函数可以重载,且构造函数是编译器在创建类的对象时根据初始化的数据自动调用的。

1. 使用构造函数

下面使用构造函数纠正 Student 类的最后一个错误。

cpp 复制代码
// 头文件
#include <iostream>
#include <string>

// 常量声明
const int SIZE_NAME = 20;
const int SIZE_ID = 11;

// 使用声明
using std::cout;
using std::endl;

// Student 类
class Student
{
private:
	char _name[SIZE_NAME];
	size_t _age;
	char _ID[SIZE_ID];

public:
	// 构造函数
	Student(const char* name, size_t age, const char* ID)
	{
		strcpy(_name, name);
		_age = age;
		strcpy(_ID, ID);
	}
	// 打印信息
	void Print()
	{
		cout << _name << " " << _age << " " << _ID << endl;
	}
};

int main()
{
	// 自动调用构造函数
	Student zhangsan("zhangsan", 18, "2101240002");
	// 打印信息
	zhangsan.Print();

	return 0;
}

程序的运行结果如下:

虽然上述程序可以正常运行,但是仍然存在 bug。如:Student zhangsan;,这条语句没有对应的构造函数,因为创建该对象是没有传递参数。

如上述这种创建参数时不显式传递参数,编译器实际上会去寻找默认构造函数。

2. 默认构造函数

默认构造函数就是不需要传递参数的构造函数,可以是全缺省,也可以是无参函数。

当用户未定义任何构造函数时,编译器会生成一个默认构造函数,该函数没有参数,且该函数不对内置类型(int、double等)进行处理,对自定义类型调用其默认构造函数。

默认构造函数有以下三种:编译器自动生成的默认构造函数、无参构造函数和全缺省构造函数。但是这三种构造函数只能存在一种。

下面是使用了默认构造函数之后的 Student 类。

cpp 复制代码
// 头文件
#include <iostream>
#include <string>

// 常量声明
const int SIZE_NAME = 20;
const int SIZE_ID = 11;

// 使用声明
using std::cout;
using std::endl;

// Student 类
class Student
{
private:
	char _name[SIZE_NAME];
	size_t _age;
	char _ID[SIZE_ID];

public:
	// 默认构造函数
	Student()
	{
		strcpy(_name, "小明");
		_age = 18;
		strcpy(_ID, "2101240001");
	}
	// 构造函数
	Student(const char* name, size_t age, const char* ID)
	{
		strcpy(_name, name);
		_age = age;
		strcpy(_ID, ID);
	}
	// 打印信息
	void Print()
	{
		cout << _name << " " << _age << " " << _ID << endl;
	}
};

int main()
{
	// 默认构造函数
	Student xiaoming;
	xiaoming.Print();
	// 构造函数
	Student zhangsan("张三", 19, "2101240002");
	zhangsan.Print();

	return 0;
}

程序的运行结果如下:

在创建小明这个对象的时候,虽然看着像创建了一个未初始化的局部变量,但是实际上编译器为其调用了默认构造函数。

七、类作用域

类有属于自己的作用域,和命名空间 namespace 类似。可以在类中定义与外部全局函数相同的函数名。

跟命名空间 namespace 一个理解就好了,相当于创建了一个新的作用域。只要不再同一个作用域,名称相同就没有问题。当然,函数重载不算。

由于类中是一个新的作用域,所以类中的成员函数在类外定义时,需要显示声明该函数所属的类域。

八、类中函数的声明和定义

如果在类中直接定义函数,那么相当于内联函数(inline),也就是在函数前面加了个关键字 inline。当然也只是给编译器一个建议,并不是直接就是内联函数。

所以通常都是在类中声明函数,在类外定义函数,而在类外定义函数又有所不同。如果在类内声明函数,在类外定义函数,那么在类外定义函数时需要指明该函数属于哪个类,否则编译器会当作全局函数处理。

九、类的实例化

使用 class 创建一个类时,实际上只是该类的声明,并没有创建空间。参照结构体,使用 struct 创建一个结构体时,只是告诉编译器我要创建一个类型,该类型包含如下内容。然后使用该类型创建具体的变量时,编译器才会开辟空间。

所以当使用类创建具体的对象时,编译器才会分配具体的空间。该过程也叫做类的实例化。

十、类的大小计算

类大小的计算和结构体类似,也存在内存对其。但是计算类大小的时候并不包括其中的函数,因为每个类的实例化对象都有其各自的成员变量数据,但是调用的都是同一个函数,所以类的函数保存在公共的代码段,所有类的实例化对象均可共享。

cpp 复制代码
// 头文件
#include <iostream>

// 使用声明
using std::cout;
using std::endl;

// Date 类声明
class Date
{
private:
	size_t _year;
	size_t _month;
	size_t _day;

public:
	Date(size_t year = 1949, size_t month = 10, size_t day = 1);  // 默认构造函数
	void Print();
};

// Date 类成员函数定义
Date::Date(size_t year, size_t month, size_t day)
{
	_year = year;
	_month = month;
	_day = day;
}

void Date::Print()
{
	cout << _year << "-" << _month << "-" << _day << endl;
}

int main()
{
	// 计算 Date 类的大小
	cout << sizeof(Date) << endl;

	return 0;
}

代码的运行结果如下:

由于我的编译器上 size_t 是 64 位的整数,所以结果算出来是 24 个字节。

上面的内存计算还是很简单的,大家自己复习一下内存对其相关的知识。

相关推荐
用余生去守护7 分钟前
python报错系列(16)--pyinstaller ????????
开发语言·python
yuanbenshidiaos10 分钟前
c++---------数据类型
java·jvm·c++
数据小爬虫@11 分钟前
利用Python爬虫快速获取商品历史价格信息
开发语言·爬虫·python
向宇it14 分钟前
【从零开始入门unity游戏开发之——C#篇25】C#面向对象动态多态——virtual、override 和 base 关键字、抽象类和抽象方法
java·开发语言·unity·c#·游戏引擎
莫名其妙小饼干30 分钟前
网上球鞋竞拍系统|Java|SSM|VUE| 前后端分离
java·开发语言·maven·mssql
十年一梦实验室39 分钟前
【C++】sophus : sim_details.hpp 实现了矩阵函数 W、其导数,以及其逆 (十七)
开发语言·c++·线性代数·矩阵
taoyong00143 分钟前
代码随想录算法训练营第十一天-239.滑动窗口最大值
c++·算法
最爱番茄味1 小时前
Python实例之函数基础打卡篇
开发语言·python
这是我581 小时前
C++打小怪游戏
c++·其他·游戏·visual studio·小怪·大型·怪物
fpcc1 小时前
跟我学c++中级篇——C++中的缓存利用
c++·缓存