【C++】类和对象(上)

简介: 【C++】类和对象(上)

面向对象和面向过程


C语言是面向过程的,关注的是过程,分析出求解问题的步骤,通过函数调用逐步解决问题。

C++是基于面向对象的,关注的是对象,将一件事情拆分成不同的对象,靠对象之间的交互完成

例如洗衣服:C语言关注的是过程。

da1da2a2955e6ee163577650daa116ba.png

C语言的思想

cfda2d3883e9ef5b0cfd114e23e7f034.png

C++的思想

这就是面向对象和面向过程的区别



C语言结构体中只能定义变量,在C++中,结构体内不仅可以定义变量,也可以定义函数。比如:之前在数据结构初阶中,用C语言方式实现的栈,结构体中只能定义变量;现在以C++方式实现,会发现struct中也可以定义函数。

typedef int DataType;
struct Stack
{
  void Init(size_t capacity)
  {
    _array = (DataType*)malloc(sizeof(DataType) * capacity);
    if (nullptr == _array)
    {
      perror("malloc申请空间失败");
      return;
    }
    _capacity = capacity;
    _size = 0;
  }
  void Push(const DataType& data)
  {
    // 扩容
    _array[_size] = data;
    ++_size;
  }
  DataType Top()
  {
      return _array[_size - 1];
  }
  void Destroy()
  {
    if (_array)
    {
      free(_array);
      _array = nullptr;
      _capacity = 0;
      _size = 0;
    }
  }
  DataType* _array;
  size_t _capacity;
  size_t _size;
};


上面的结构体定义在C++中更喜欢用class来代替。


1.类的定义

class className
{
  //类体(由成员函数和变量组成)  
};//注意这里有分号


class为定义类的关键字ClassName为类的名字,{}中为类的主体,注意类定义结束时后面分号不能省略

类体中内容称为类的成员:类中的变量称为类的属性或成员变量; 类中的函数称为类的方法或者成员函数


2.类的两种定义方式


1.1 成员函数声明和定义都放在类中

注意:成员函数如果在类中定义,编译器可能会将其当作内联函数处理。

496cef57f9f88211cf3ccaf0eff37153.png


1.2 声明和定义分离

一般在项目中,声明和定义都是分离的,之所以这么做,有两种原因,其一就是方便阅读,其二就是为了防止源码泄露。所以我们以后尽量都使用这种。

bf11738fd1b8fa692dc777ef27eb3418.png

3.成员变量的命名建议

我们看下面一个例子

//假设我们要定义一个日期类
class Date
{
public:
  void Init(int year, int month, int day)//在初始化的过程中就会出现这种不方便区分的情况,虽然语法上没有问题。
    {
        year = year;
        month = month;
        day = day;
    }
private:
  int year;
    int month;
    int day;
};


所以类中成员变量定义的时候,我们一般在命名时使用一些区分手段,比如定义成"_year,_month,_day"或者"year_,\month_,day_"或者"mYear,mMonth,mDay"等,一般这种命名规则时看公司的要求,加个前缀或者后缀以示区分就可以了。


4.类的访问限定符和封装

访问限定符

C++实现封装的方式:用类将对象的属性与方法结合在一块,让对象更加完善,通过访问权限选择性的将其接口提供给外部的用户使用。

这个选择性就是通过访问限定符实现的。

ffa360e35ba092c72029d8946f47466e.png

  1. 其中public修饰的成员可以在类外面直接访问;
  2. protected和private在类外不能被直接访问(关于protected和private的区别我们以后再说);
  3. 访问权限的作用域从该访问限定符出现开始到下一个访问限定符出现或者类结束为止;
  4. class和struct的区别:class默认访问权限为private,struct的默认访问权限是public(这是为了兼容C语言);


注意:访问限定符只在编译时有用,当数据映射到内存后,没有任何访问限定符上的区别

封装

封装:将数据和操作数据的方法进行有机结合,隐藏对象的属性和实现细节,仅对外公开接口来和对象进行交互。

封装本质上是一种管理,让用户更方便使用类。比如:对于电脑这样一个复杂的设备,提供给用户的就只有开关机键、通过键盘输入,显示器,USB插孔等,让用户和计算机进行交互,完成日常事务。但实际上电脑真正工作的却是CPU、显卡、内存等一些硬件元件。对于计算机使用者而言,不用关心内部核心部件,比如主板上线路是如何布局的,CPU内部是如何设计的等,用户只需要知道,怎么开机、怎么通过键盘和鼠标与计算机进行交互即可。因此计算机厂商在出厂时,在外部套上壳子,将内部实现细节隐藏起来,仅仅对外提供开关机、鼠标以及键盘插孔等,让用户可以与计算机进行交互即可。

在C++语言中实现封装,可以通过类将数据以及操作数据的方法进行有机结合,通过访问权限来隐藏对象内部实现细节,控制哪些方法可以在类外部直接被使用。

5.类的作用域

类定义了一个新的作用域,所有成员都在这个作用域中,在类外定义类的成员的时候,需要使用::作用域操作符指明成员属于哪个类。

class Stack
{
public:
    void Init(int N = 4);
private:
    int* _a;
    int _size;
    int _capacity;
};
void Stack::Init(int N)//这里要使用::
{
    //实现函数
    //...
}


6.类的实例化

class Date
{
public:
    Init();
    //...
private:
  int _year;
    int _month;
    int _day;
};
//上面我们定义了一个类
int main()
{
    Date.Init();//这是错误的
    return 0;
}


对于上述的错误,我们可以类比C语言中的如下错误

struct Stack
{
  int* a;
    int size;
    int capacity;
};
int main()
{
    Stack.size = 0;//这里也是错误
    return 0;
}


这中错误本质上就是我们只定义了类和结构体,他们本身在内存中没有位置,就是不占用内存,变量是不存在的,需要使用其定义变量,然后对变量进行操作。打个比方,定义的类和结构体就是建房子使用的蓝图不占用空间,定义的变量就是建好的房子,是占用空间的。


在C++中,这种使用类定义变量的操作叫做类的实例化,实例化出来的变量叫做对象,一个类可以实例化出多个对象。

3b4f744bd5a372e686c80a6004765cfc.png

7.类对象模型

7.1类对象的存储方式

类比C语言,类的属性,也就相当于结构体的成员,所以存储方式和C语言是相同的,但是C语言中并没有成员函数,我们无法对标,那么成员函数的存储的位置在哪里呢?


我们实例化类之后会创建多个对象,这些通过同一个类实例化出来的对象的方法都是相同的,所以调用的是同一个函数,所以只需要存储一个函数地址即可,C++的实现方式就是对象中只存放成员变量,成员函数放在一个公共的代码区,对象使用的时候直接调用即可。如下图:

5a95ba4bf66136fc6453546af1e85d7c.png

7.2类对象的大小计算

  1. 类中既有成员变量又有成员函数的时候:由于成员函数存放在公共的代码区,所以对象的大小不计算成员函数,只计算成员变量,和C语言中的结构体的计算方法相同,都需要考虑内存对齐
  2. 类中只有成员变量或者类是空类的时候,需要一个字节的大小用于占位,证明这个类的存在,否则我们使用这个类实例化对象的时候,就会出现问题。


8.this指针

首先我们在这里定义一个日期类(Date),请记住这个类,我们以后会经常用到它

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;
};


现在我们看下面一段代码和运行结果:

98fdc4255b6c8c5acbf72a1332e412f0.png

这时候就会有一个疑问:为什么d1和d2调用的都是同一个Init和Print,但是执行完之后的结果却不相同呢?


这是因为在函数中有一个隐藏的变量,这个变量就是this(这个this的类型是Date* const ,用于保护this存放的地址不变),在函数调用的时候,编译器会自动传入一个参数,就是对象的地址。


this指针的一些特性

  1. this指针的类型:类类型 const*,即成员函数中,不能给this指针赋值。
  2. 只能在“成员函数”的内部使用
  3. this指针本质上是“成员函数”的形参,当对象调用成员函数时,将对象地址作为实参传递给this形参。所以对象中不存储this指针。
  4. this指针是“成员函数”第一个隐含的指针形参,一般情况由编译器通过ecx寄存器自动传递,不需要用户传递


注意:我们不能在传参的时候自己传对象的地址,不能在成员函数定义的过程中显示的写出this这个参数,这些都是靠编译器自动完成的,但是我们在成员函数中可以使用this这个变量。

s指针赋值。

  1. 只能在“成员函数”的内部使用
  2. this指针本质上是“成员函数”的形参,当对象调用成员函数时,将对象地址作为实参传递给this形参。所以对象中不存储this指针。
  3. this指针是“成员函数”第一个隐含的指针形参,一般情况由编译器通过ecx寄存器自动传递,不需要用户传递


注意:我们不能在传参的时候自己传对象的地址,不能在成员函数定义的过程中显示的写出this这个参数,这些都是靠编译器自动完成的,但是我们在成员函数中可以使用this这个变量。

相关文章
|
1天前
|
编译器 C语言 C++
|
1天前
|
编译器 C++
【C++】详解初始化列表,隐式类型转化,类静态成员,友元
【C++】详解初始化列表,隐式类型转化,类静态成员,友元
|
4天前
|
存储 编译器 C++
【C++】类和对象④(再谈构造函数:初始化列表,隐式类型转换,缺省值
C++中的隐式类型转换在变量赋值和函数调用中常见,如`double`转`int`。取引用时,须用`const`以防修改临时变量,如`const int& b = a;`。类可以有隐式单参构造,使`A aa2 = 1;`合法,但`explicit`关键字可阻止这种转换。C++11起,成员变量可设默认值,如`int _b1 = 1;`。博客探讨构造函数、初始化列表及编译器优化,关注更多C++特性。
|
4天前
|
编译器 C++
【C++】类和对象④(类的默认成员函数:取地址及const取地址重载 )
本文探讨了C++中类的成员函数,特别是取地址及const取地址操作符重载,通常无需重载,但展示了如何自定义以适应特定需求。接着讨论了构造函数的重要性,尤其是使用初始化列表来高效地初始化类的成员,包括对象成员、引用和const成员。初始化列表确保在对象创建时正确赋值,并遵循特定的执行顺序。
|
4天前
|
C语言 C++
【C++】日期类Date(详解)③
该文介绍了C++中直接相减法计算两个日期之间差值的方法,包括确定max和min、按年计算天数、日期矫正及计算差值。同时,文章讲解了const成员函数,用于不修改类成员的函数,并给出了`GetMonthDay`和`CheckDate`的const版本。此外,讨论了流插入和流提取的重载,需在类外部定义以符合内置类型输入输出习惯,并介绍了友元机制,允许非成员函数访问类的私有成员。全文旨在深化对运算符重载、const成员和流操作的理解。
|
4天前
|
C++
【C++】日期类Date(详解)②
- `-=`通过复用`+=`实现,`Date operator-(int day)`则通过创建副本并调用`-=`。 - 前置`++`和后置`++`同样使用重载,类似地,前置`--`和后置`--`也复用了`+=`和`-=1`。 - 比较运算符重载如`&gt;`, `==`, `&lt;`, `&lt;=`, `!=`,通常只需实现两个,其他可通过复合逻辑得出。 - `Date`减`Date`返回天数,通过迭代较小日期直到与较大日期相等,记录步数和符号。 ``` 这是236个字符的摘要,符合240字符以内的要求,涵盖了日期类中运算符重载的主要实现。
|
4天前
|
定位技术 C语言 C++
C++】日期类Date(详解)①
这篇教程讲解了如何使用C++实现一个日期类`Date`,涵盖操作符重载、拷贝构造、赋值运算符及友元函数。类包含年、月、日私有成员,提供合法性检查、获取某月天数、日期加减运算、比较运算符等功能。示例代码包括`GetMonthDay`、`CheckDate`、构造函数、拷贝构造函数、赋值运算符和相关运算符重载的实现。
|
4天前
|
编译器 C++
【C++】类和对象③(类的默认成员函数:赋值运算符重载)
在C++中,运算符重载允许为用户定义的类型扩展运算符功能,但不能创建新运算符如`operator@`。重载的运算符必须至少有一个类类型参数,且不能改变内置类型运算符的含义。`.*::sizeof?`不可重载。赋值运算符`=`通常作为成员函数重载,确保封装性,如`Date`类的`operator==`。赋值运算符应返回引用并检查自我赋值。当未显式重载时,编译器提供默认实现,但这可能不足以处理资源管理。拷贝构造和赋值运算符在对象复制中有不同用途,需根据类需求定制实现。正确实现它们对避免数据错误和内存问题至关重要。接下来将探讨更多操作符重载和默认成员函数。
|
4天前
|
存储 编译器 C++
【C++】类和对象③(类的默认成员函数:拷贝构造函数)
本文探讨了C++中拷贝构造函数和赋值运算符重载的重要性。拷贝构造函数用于创建与已有对象相同的新对象,尤其在类涉及资源管理时需谨慎处理,以防止浅拷贝导致的问题。默认拷贝构造函数进行字节级复制,可能导致资源重复释放。例子展示了未正确实现拷贝构造函数时可能导致的无限递归。此外,文章提到了拷贝构造函数的常见应用场景,如函数参数、返回值和对象初始化,并指出类对象在赋值或作为函数参数时会隐式调用拷贝构造。
|
4天前
|
存储 编译器 C语言
【C++】类和对象②(类的默认成员函数:构造函数 | 析构函数)
C++类的六大默认成员函数包括构造函数、析构函数、拷贝构造、赋值运算符、取地址重载及const取址。构造函数用于对象初始化,无返回值,名称与类名相同,可重载。若未定义,编译器提供默认无参构造。析构函数负责对象销毁,名字前加`~`,无参数无返回,自动调用以释放资源。一个类只有一个析构函数。两者确保对象生命周期中正确初始化和清理。