[C++] 轻熟类和对象

DevKevin 2024-07-24 15:35:01 阅读 100

Kevin的技术博客.png

类的定义

格式规范

<code>class为定义类的关键字,后有类名,类的主体存于{}中;类定义结束时后面的分号不能省略;类体的内容成为类的成员,类中的变量成为成员变量,函数成为方法或成员函数;C++兼容C语言的struct用法,同时将struct升级成了类的用法(更推荐类)在类定义中直接定义一个成员函数(即在类声明的花括号{}内直接给出函数体),编译器会默认将这个成员函数视为inline。这样做的好处是,如果成员函数非常简单,编译器可以优化代码,通过内联展开来提高程序的执行效率。

举例代码:

class Stack

{

public:

// 成员函数

void Init(int n = 4)

{

array = (int*)malloc(sizeof(int) * n);

if (nullptr == array)

{

perror("malloc申请空间失败");

return;

}

capacity = n;

top = 0;

private:

// 成员变量

int* array;

size_t capacity;

size_t top;

}; // 分号不能省略

访问限定符

封装的概念

封装是OOP的一个基本原则,它指的是将数据(属性)和操作这些数据的代码(方法)组合在一起,形成一个单元。封装的主要目的是隐藏内部实现的细节,只暴露出一个可以被外界访问和操作的接口。

数据封装:类定义了属性(也称为成员变量或字段),这些属性代表了对象的状态。封装确保了这些属性只能通过类提供的方法来访问和修改,从而保护数据不被外部代码直接访问,避免数据被不当操作。方法封装:类还定义了方法(也称为成员函数或行为),这些方法定义了可以对对象执行的操作。封装确保了对象的行为是通过这些方法来实现的,而不是直接操作对象的内部状态。

访问限定符

image.png

隐藏实现细节:通过使用private访问限定符,类的实现细节被隐藏起来,外部代码不能直接访问或修改对象的内部状态,只能通过public方法来操作。提供接口:public方法提供了一个接口,允许外部代码以受控的方式与对象交互。这些方法可以包含对private成员的访问和修改,但这些操作的细节对外部是不可见的。继承和多态性:protected访问限定符允许子类访问和修改父类的某些成员,这在实现继承和多态性时非常有用。子类可以扩展或修改父类的行为,同时保持对外部代码的封装。维护数据完整性:通过限制对属性的直接访问,封装确保了数据的完整性和一致性。促进模块化:封装使得代码更加模块化,每个类负责管理自己的数据和行为,减少了不同模块之间的耦合。(高内聚、低耦合)

类域

类会定义一个新的作用域,类的所有成员都在类的作用域中,在类外定义成员时需要用<code>::作用域操作符指明成员属于哪个类域。

在类外定义成员函数:

#include<iostream>

using namespace std;

class Stack

{

public:

// 成员函数

void Init(int n = 4);

private:

// 成员变量

int* array;

size_t capacity;

size_t top;

};

// 声明和定义分离,需要指定类域

void Stack::Init(int n)

{

array = (int*)malloc(sizeof(int) * n);

if (nullptr == array)

{

perror("malloc申请空间失败");

return;

}

capacity = n;

top = 0;

}

实例化

概念

类是一个蓝图或模板,它定义了一组属性(成员变量)和方法(成员函数),这些属性和方法共同描述了一类事物的特征和行为。而通过创建对象就可以将对象进行实例化,这一一种一对多的关系,一个类可以创造多个对象,每一个类都是一个个体实例,并不冲突。

<code>// 如何实例化对象

// 定义一个日期类

class Date

{

public:

void Init(int year, int month, int day)

{

_year = year;

_month = month;

_day = day;

}

void Print()

{

cout << _year << "/" << _month << "/" << _day << endl;

}

private:

// 这⾥只是声明,没有开空间

int _year;

int _month;

int _day;

};

int main()

{

// Date类实例化出对象d1和d2

Date d1;

Date d2;

d1.Init(2024, 3, 31);

d1.Print();

d2.Init(2024, 7, 5);

d2.Print();

return 0;

}

对象⼤⼩

对象中只存储成员变量,每个成员函数在对象调用的时候用的都是同一个指针来访问。对于对象的大小,要符合内存对齐的规则,关于内存对齐的详细运用理解可以点击蓝字阅读我的另一篇博客,具体的规则大概如下:

• 第⼀个成员在与结构体偏移量为0的地址处。

• 其他成员变量要对⻬到某个数字(对⻬数)的整数倍的地址处。

• 注意:对⻬数 = 编译器默认的⼀个对⻬数 与 该成员⼤⼩的较⼩值。

• VS中默认的对⻬数为8

• 结构体总⼤⼩为:最⼤对⻬数(所有变量类型最⼤者与默认对⻬参数取最⼩)的整数倍。

• 如果嵌套了结构体的情况,嵌套的结构体对⻬到⾃⼰的最⼤对⻬数的整数倍处,结构体的整体⼤⼩ 就是所有最⼤对⻬数(含嵌套结构体的对⻬数)的整数倍。

•不要忘记内存对齐的目的:为了减少CPU访问内存次数,提高效率。

this指针

编译器编译后,类的成员函数默认都会在形参第⼀个位置,增加⼀个当前类类型的指针,叫做<code>this指针。⽐如Date类的Init的真实原型为, void Init(Date* const this, int year, int month, int day),但是C++规定不能在实参和形参位置上写this指针,但是可以在函数体内显式使用this指针

类的成员函数中访问成员变量,本质都是通过this指针访问的,如Init函数中给_year赋值, this->_year = year;。所以通过this指针可以对于一个对象的维护更加便捷。

// 显式使用this指针

void Init(int year, int month, int day)

{

_year = year;

this->_month = month;

this->_day = day;

}

存储位置

关于this指针在内存上的存储位置:当成员函数被调用时,this 指针通常存储在调用栈上。调用栈是用于存储函数调用时的局部变量和状态信息的内存区域。

image.png



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。