C++类和对象概念及实现详解(上篇)

news2025/1/10 18:39:31

文章目录

一、什么是类和对象呢?

1、类的引入

2、类的定义

3、类的访问限定符

4、类对象的储存方式

5、this指针的特性

二、类的六个默认成员函数详解

1、构造函数

2、析构函数

3、未完待续…… 


标题:类和对象概念及实现详解(上篇)

作者:@Ggggggtm

寄语:与其忙着诉苦,不如低头赶路,奋路前行,终将遇到一番好风景

 

一、什么是类和对象呢?

1、类的引入

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

2、类的定义

class className
{
// 类体:由成员函数和成员变量组成
};  // 一定要注意后面的分号

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

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

   类的两种定义方式:

  1. 声明和定义全部放在类体中,需注意:成员函数如果 在类中定义 ,编译器可能会将其当成 联函数 处理。
  2. 类声明放在 .h 文件中,成员函数定义放在 .cpp 文件中,注意: 成员函数名前需要加类名 ::。 

3、类的访问限定符

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

访问限定符说明:

  1. public修饰的成员在类外可以直接被访问;
  2. protectedprivate修饰的成员在类外不能直接被访问
  3. 访问权限作用域从该访问限定符出现的位置开始直到下一个访问限定符出现时为止;
  4. 如果后面没有访问限定符,作用域就到 } 即类结束;
  5. class的默认访问权限为privatestructpublic(因为struct要兼容C) 。

4、类对象的储存方式

  我们假想:每个对象中成员变量是不同的,但是调用同一份成员函数,如果按照每实例一个对象都给成员变量和成员函数创造一次空间存储,当一 个类创建多个对象时,每个对象中都会保存一份代码,相同代码保存多次,浪费空间。那么如何解决呢? 

  针对上面的问题,类的存储就变成了:只保存成员变量,成员函数存放在公共的代码段 。那么一个类的大小其实就是:实际就是该类中成员变量之和,当然要注意内存对齐,注意空类的大小,空类比较特殊,编译器给了空类一个字节来唯一标识这个类的对象。

5、this指针的特性

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;
     d1.Init(2022,1,11);
     d2.Init(2022, 1, 12);
     d1.Print();
     d2.Print();
     return 0; 
}

  我们知道了成员函数是放在了公共代码段。函数体中没有关于不同对象的区分。那么在上面的代码中d1和d2同时掉用了Print()函数,怎么是分别打印出d1对象中的成员变量和d2对象中的成员变量呢?(当然Init函数与Print函数的区分类似)。

  C++ 中通过引入 this 指针解决该问题,即: C++ 编译器给每个 非静态的成员函数 增加了一个隐藏 的指针参数,让该指针指向当前对象 ( 函数运行时调用该函数的对象 ) ,在函数体中所有 成员变量 的操作,都是通过该指针去访问。只不过所有的操作对用户是透明的,即用户不需要来传递,编 译器自动完成 。如下图:

 

this指针的特性:

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

二、类的六个默认成员函数详解

  什么是默认成员函数呢?

  默认成员函数:用户没有显式实现,编译器会生成的成员函数称为默认成员函数。

  如果一个类中什么成员都没有,简称为空类。 空类中真的什么都没有吗?并不是,任何类在什么都不写时,编译器会自动生成以下6 个默认成员函数
  1. 构造函数;
  2. 析构函数;
  3. 拷贝构造;
  4. 赋值重载;
  5. 普通对象取地址;
  6. const对象取地址。

  我们来看一下各个默认的成员函数的概念及实现。本篇我们先掌握构造函数和析构函数,这两个时相对较为麻烦和重要的,下篇我们会接着是西安剩余的默认成员函数以及类和对象剩余的重要的部分。

1、构造函数

  构造函数是一个特殊的成员函数,名字与类名相同,创建类类型对象时由编译器自动调用,以保证每个数据成员都有 一个合适的初始值,并且在对象整个生命周期内只调用一次。

  构造函数是特殊的成员函数,需要注意的是,构造函数虽然名称叫构造,但是构造函数的主要任务并不是开空间创建对象,而是初始化对象其特征如下:

  • 函数名与类名相同;
  • 无返回值;
  • 对象实例化时自动调用对应的构造函数;
  • 构造函数可以重载。
  • 如果如果类中没有显式定义构造函数,则C++编译器会自动生成一个默认的无参构造函数,一但用户显式定义编译器将不再生成。

  我们结合着以下代码一起理解以下。

  class Date
 {
  public:
      // 1.无参构造函数
      Date()
     {}
  
      // 2.带参构造函数
      Date(int year, int month, int day)
     {
          _year = year;
          _month = month;
          _day = day;
     }
  private:
      int _year;
      int _month;
      int _day;
 };
  
  void TestDate()
 {
      Date d1; // 调用无参构造函数
      Date d2(2015, 1, 1); // 调用带参的构造函数
  
      // 注意:如果通过无参构造函数创建对象时,对象后面不用跟括号,否则就成了函数声明
      // 以下代码的函数:声明了d3函数,该函数无参,返回一个日期类型的对象
      // warning C4930: “Date d3(void)”: 未调用原型函数(是否是有意用变量定义的?)
      Date d3();
 }
  关于编译器生成的默认成员函数,很多人都会有疑惑:不实现构造函数的情况下,编译器会
生成默认的构造函数。但是看起来默认构造函数又没什么用? 象调用了编译器生成的默
认构造函数,但是 对象中的成员函数 依旧是随机值。也就说在这里 编译器生成的 默认构造函数并没有什么用??
  C++ 把类型分成内置类型 ( 基本类型 ) 和自定义类型。内置类型就是语言提供的数据类
型,如: int/char... ,自定义类型就是我们使用 class/struct/union等自己定义的类型。其实编译器生成默认的构造函数会对自定类型成员 调用的它的默认成员函数,并不会对内置类型的成员变量进行初始化。 
  如下面代码:Data中并没有自己实现构造函数,系统会自动生成。让后会对自定义类型成员_t调用它的默认成员函数。
class Time
{
public:
 Time()
 {
     cout << "Time()" << endl;
     _hour = 0;
     _minute = 0;
     _second = 0;
 }
private:
     int _hour;
     int _minute;
     int _second;
};
class Date
{
private:
     // 基本类型(内置类型)
     int _year;
     int _month;
     int _day;
     // 自定义类型
     Time _t;
};
int main()
{
     Date d;
     return 0; 
}
  无参的构造函数和全缺省的构造函数都称为默认构造函数,并且默认构造函数只能有一个。
注意:无参构造函数、全缺省构造函数、我们没写编译器默认生成的构造函数,都可以认为
是默认构造函数。

2、析构函数

  通过前面构造函数的学习,我们知道一个对象是怎么来的,那一个对象又是怎么没呢的?这里就用到了析构函数。
  析构函数:与构造函数功能相反,析构函数不是完成对对象本身的销毁,局部对象销毁工作是由编译器完成的。而对象在销毁时会自动调用析构函数,完成对象中资源的清理工作
析构函数 是特殊的成员函数,其 特征 如下:
  •  析构函数名是在类名前加上字符 ~
  • 无参数无返回值类型;
  • 一个类只能有一个析构函数。若未显式定义,系统会自动生成默认的析构函数。注意:析构函数不能重载;
  • 对象生命周期结束时, C++ 编译系统系统自动调用析构函数。

 我们结合下面代码一起理解一下。

typedef int DataType;
class Stack
{
public:
 Stack(size_t capacity = 3)
 {
     _array = (DataType*)malloc(sizeof(DataType) * capacity);
     if (NULL == _array)
     {
         perror("malloc申请空间失败!!!");
         return;
     }
     _capacity = capacity;
     _size = 0;
}
void Push(DataType data)
{
     //CheckCapacity();
     _array[_size] = data;
     _size++;
}
 // 其他方法...
 ~Stack()
 {
     if (_array)
     {
         free(_array);
         _array = NULL;
         _capacity = 0;
         _size = 0;
     }
}
private:
     DataType* _array;
     int _capacity;
     int _size;
};
void TestStack()
{
     Stack s;
     s.Push(1);
     s.Push(2);
}
  关于编译器自动生成的析构函数,是否会完成一些事情呢?编译器生成的默认析构函数,对自定类型成员调用它的析构函数。 注意:创建那个类的对象则调用该类的析构函数,销毁哪个类的对象则调用该类的析构函数。
  如果类中没有申请资源时,析构函数可以不写,直接使用编译器生成的默认析构函数, 有资源申请时,一定要写,否则会造成资源泄漏。

3、未完待续…… 

  今天的内容就到这里,需要重点理解和掌握构造函数和析构函数,后续我会更新下篇带大家理解完类和对象ovo!

  希望以上内容对你有所帮助,感谢阅读。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/114446.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

vue3 antd table表格——自定义单元格样式(二)利用rowClassName给table添加行样式

vue3 antd项目实战——修改ant design vue组件中table表格的默认样式&#xff08;二&#xff09;知识调用场景复现修改table表格的行样式一、rowClassName添加行样式二、表格的不可控操作写在最后知识调用 文章中可能会用到的知识链接vue3ant design vuets实战【ant-design-vu…

从头开始用树莓派做一个NAS【最新超详细教程】

一、概述 众所周知在办公的时候两台电脑之间经常倒数据资料非常麻烦&#xff0c;而NAS可以很好的解决这个问题。树莓派搭建NAS方法有很多&#xff0c;我们之前也拍过直接用Samba、FTP这些来实现NAS功能&#xff0c;但是这些需要你会在命令行进行配置&#xff0c;而且对于新手用…

【Linux】Linux权限管理

目录一.Linux用户权限1.权限的概念2.用户分类3.切换用户4.sudo提权二.Linux文件权限1.文件属性2.文件类型3.文件角色划分4.基本权限三.文件访问权限的相关设置方法1.chmod2.chown3.charp4.file5.权限拒绝四.默认权限umask五.目录的权限六.粘滞位1.背景2.准备3.情况4.粘滞位一.L…

初识Docker:(1)什么是docker

初识Docker&#xff1a;&#xff08;1&#xff09;什么是docker项目部署的问题Docker总结项目部署的问题 大型项目组件较多&#xff0c;运行环境也较为复杂&#xff0c;部署时会碰到一些问题&#xff1a; 依赖关系复杂&#xff0c;容易出现兼容性问题开发、测试、生产环境有差…

git revert以及revert的恢复

一&#xff1a;背景与方案 在工作中遇见的这样的场景&#xff1a; 场景一&#xff1a; 已经merge到待发布的版本分支中的功能需要移除当前的分支&#xff0c;改在后续版本发布&#xff0c;示意图如下&#xff0c;展示的是commit序列&#xff0c; 这里想要移除的功能是commi…

[python库] base64库的基本使用

1. base64是什么 base64是一种二进制到文本格式的编码方式。具体来说就是将byte数组编码为字符串的方法&#xff0c;而编码出来的字符串只包含ASCII基础字符。 虽然说base64是一种编码方式&#xff0c;但是它并不推荐作为常规的加密算法使用&#xff0c;因为该算法的加解密算法…

Android开发进阶——binder通讯学习

什么是binder 通常意义下&#xff0c;binder指的是一种通信机制对Server端来说&#xff0c;Binder指的是Binder本地对象&#xff0c;对于Client端来说&#xff0c;Binder指的是Binder代理对象对于传输过程而言&#xff0c;binder是可以跨进程传输的对象 Binder的基本原理 Bi…

【工作流Activiti7】7、Activiti7+SpringBoot

1. 版本问题 1.1. Activiti版本 7.1.0-M6是最后一个支持JDK1.8的版本&#xff0c;此后的版本都要求JDK11以上 目前&#xff0c;Activiti最新版本是7.6.0&#xff0c;它是用JDK11编译的&#xff0c;因此要想使用最新版7.6.0必须升级JDK版本&#xff0c;不能再用1.8 同时&…

【数组中数字出现的次数-有限状态自动机】

数组中数字出现的次数一&#xff0c;有限状态自动机解法二&#xff0c;一般解法想必大家对数组中数字出现的次数的这种题并不少见&#xff0c; 主要有三种&#xff1a; 1&#xff0c;找出数组中只出现一次的数字&#xff08;其他数字出现两次&#xff09; 2&#xff0c;找出数组…

渗透测试指操作系统漏洞发现与防御概述

今天继续给大家介绍渗透测试相关知识&#xff0c;本文主要内容是渗透测试指操作系统漏洞发现与防御概述。 免责声明&#xff1a; 本文所介绍的内容仅做学习交流使用&#xff0c;严禁利用文中技术进行非法行为&#xff0c;否则造成一切严重后果自负&#xff01; 再次强调&#x…

Android四大组件之Service

文章目录Android四大组件之默默劳动的Service什么是ServiceAndroid多线程编程线程的基本用法在子线程中更新UI解析异步消息处理机制MessageHandlerMessageQueueLooper异步消息的整个流程使用AsyncTaskService的基本用法定义一个Service启动和停止ServiceActivity和Service进行通…

【4】axi协议学习

1、axi背景介绍: Advanced extensible Interface(AXI)是为了满足高性能系统设计而定义的一套独立通道协议,首次是在2003年发布的AMBA3标准中出现,经历AMBA4,目前已经到达AMBA5版本。 2、axi 特性: AXI满足如下的特性: 适合于高带宽,低延迟的设计 不需要通过复杂的桥…

去耦电容和旁路的概念说明与应用说明

回想当初第一眼看觉得&#xff0c;这啥玩意自己又菜了&#xff01; 电容&#xff1a;本质就是充放电&#xff0c;实际应该我们围着这个来转 一、解释 旁路电容从英文角度看&#xff0c;就是抄小路的意思&#xff0c;意思当有一个干扰来临时候&#xff0c;可以通过这个电容抄…

LeetCode 1739. 放置盒子:数学 思维

【LetMeFly】1739.放置盒子 力扣题目链接&#xff1a;https://leetcode.cn/problems/building-boxes/ 有一个立方体房间&#xff0c;其长度、宽度和高度都等于 n 个单位。请你在房间里放置 n 个盒子&#xff0c;每个盒子都是一个单位边长的立方体。放置规则如下&#xff1a; …

世界杯竞猜项目Dapp-第五章(合约升级)

目前主流有三种合约升级方法 transparent 方式&#xff1b;(通用&#xff0c;业务逻辑和代理逻辑解耦合&#xff0c;比较贵&#xff09;uups 方式&#xff1b;&#xff08;代理逻辑集成到了业务逻辑&#xff0c;通过继承来实现&#xff0c;便宜&#xff09;beacon 方式&#x…

408 考研《操作系统》第三章第二节:基本分页存储管理、两级页表、基本分段存储管理方式、段页式管理方式

文章目录教程1. 基本分页存储管理的基本概念1.1 连续分配方式的缺点1.2 把“固定分区分配”改造为“非连续分配版本”1.3 什么是分页存储1.4 如何实现地址的转换&#xff1f;1.5 逻辑地址结构1.6 重要的数据结构——页表1.7 知识回顾与重要考点2. 基本地址变换机构2.1 例题2.2 …

node.js+uni计算机毕设项目购物小程序(程序+小程序+LW)

该项目含有源码、文档、程序、数据库、配套开发软件、软件安装教程。欢迎交流 项目运行 环境配置&#xff1a; Node.js Vscode Mysql5.7 HBuilderXNavicat11VueExpress。 项目技术&#xff1a; Express框架 Node.js Vue 等等组成&#xff0c;B/S模式 Vscode管理前后端分离等…

不写一行代码(一):实现安卓基于GPIO的LED设备驱动

文章目录系列文章一、前言二、准备工作2.1 内核版本2.2 内核文档&#xff1a;bindings->leds2.3 文档解析&#xff1a; leds-gpio.txt三、编写DTS3.1 查原理图&#xff0c;挑选GPIO3.2 编写DTS文件四、编译测试4.1 编译dt.img4.2 烧录dt.img五、基于fs的测试5.1 测试命令5.2…

第二十五章 数论——约数

第二十五章 数论——约数一、什么是约数二、约数的求解——试除法1、问题2、思路分析3、代码实现三、约数个数1、问题描述2、算法思路3、代码实现四、约数之和1、问题描述2、算法思路3、代码实现五、最大公约数——欧几里德算法1、问题描述2、算法思路&#xff08;1&#xff09…

前端实现文件上传(点击+拖拽)

一、简介 之前在Vue项目中使用过element的上传组件&#xff0c;实现了点击上传拖拽上传的两种上传功能。然后我就在想是否可以通过原生的htmljs来实现文件的点击上传和拖拽上传&#xff0c;说干就干。 首先是点击获取上传文件自然没的说&#xff0c;只需要借助input标签即可&a…