系统学习c++类和对象——深度理解默认成员函数

前言类和对象是面向对象语言的重要概念。 c++身为一门既面向过程,又面向对象的语言。 想要学习c++, 首先同样要先了解类和对象。 本节就类和对象的几种构造函数相关内容进行深入的讲解。

目录

类和对象的基本概念

封装

类域和类体

访问限定符

private

public

protect

默认访问限定

成员函数与this指针

构造函数

初始化规则 

三个必须初始化的变量类型

默认构造函数

默认成员函数

默认构造函数

默认拷贝构造函数

拷贝构造

默认拷贝构造

析构函数

析构函数

默认析构函数


类和对象的基本概念

是一种抽象的数据类型。它规定了某种事物的特征(成员变量,可以理解为一种属性,是静态的)和行为(成员函数,可以理解为动作,是动态的)。和c语言结构体类似, 类可以看作这种事物的模板或者蓝图

对象则是类的具体实例,是具体的。拥有着类规定的特征和行为。 我们都知道, 一个蓝图可以创造许多建筑。 那么一个类也可以创建多个对象。并且每个对象都有着类似的特征和行为。 

如图就是定义的一个类,这个类规定了事物stu有三个特征:姓名, 学号, 成绩。 有着一个行为:修改成绩。并且将其实例化出对象p1, 此时p1有着这个类规定的三个特征,一个行为—— p1有着自己的名字, 有着自己的学号, 有着自己的成绩。 同时他还能修改自己的成绩。

c++的小语法相对于c语言多了许多,虽然使用方便,但是刚开始学c++也可能很难全部掌握,所以建议刚开始学习c++的时候建议勤看看小的知识点。比如c向c++过渡的一些知识点。这样学着会舒服很多, 不会有一种模糊的感觉。比如图中的成员函数的知识点就是缺省参数,这同样是c向c++过渡的一些小知识点。 但是掌握不熟悉就会迷惑。不要不信,比如我问你半缺省(int x = 10, int y )这样是否正确,你根据你心里的答案, 然后拟定传参(1), 你看看是否正确?这个1是给谁?如果你很懵, 那你可以在看完这篇文章后去复习它。恭喜你对这个知识点又加深了一点理解。 

封装

类和对象三大特征:封装, 继承和多态,这里讲解封装, 继承和多态后续再说。

首先, 封装的概念很好理解, 就是一个或多个特征属性一个或多个行为方法封装起来。只对外公布接口。使用人通过接口操作修改对象的属性。 封装有利于数据的安全性完整性同时有利于代码的模块化可维护

图中的student类封装起来了三个特征变量:name, num, grade。 一个接口函数revise_grade

图中private, public是访问限定符, 决定了类的实例是否可以直接调用类的成员。

类域和类体

类域类体是类和对象的两个重要组成部分。

类域是指类的属性和特性, 也就是类的成员变量。 类域可以是公共的(public),私有的(private)或受保护的(protected)。公共的类域可以被类的外部访问私有的类域只能在类内部访问。受保护的类域这里不解释, 可以看作和私有的类域相同。

类体是指包含在类声明中的代码块,它定义了类的行为和功能。类体包含了构造函数、成员函数、成员变量,以及其他相关代码。也就是说类体包含类域

访问限定符

访问限定符有三个: private, public, protect.

private

private的意思的私有, 在private以下定义的成员不能被类的实例直接访问到。 只能通过成员函数间接访问。 

这里可以利用成员函数revise_grade间接访问

public

public的意思是公有, public之后定义的成员可以被类的实例访问。

 

protect

protect和private类似, 在他们之后定义的成员都不可直接访问。

默认访问限定

访问限定符的范围是该访问限定符到下一个访问限定符的中间的内容。如果下面没有访问限定符, 那么就是之后的所有范围。

 struct的默认访问限定是public;

class的默认访问限定是private。

这些记住就行。

成员函数与this指针

要理解封装的概念, 还要理解this指针在类中起到的作用。

类中的成员函数默认第一个形参都是this指针, this指针是一个关键词,并且这个this指针调用该函数的对象。我们平常定义构造函数或者成员函数是看不到定义this指针知识将this指针隐藏了。

this指针可以让我们找到特定的对象

构造函数

构造函数是类和对象中新添的一种概念。类在进行实例化时会自动调用自己的构造函数。

如图红框框就是定义的一个构造函数,构造函数的定义方式与普通函数有两处不同:

1.构造函数没有返回值

2.构造函数有符号化列表,也就是图中的绿框框。

符号化列表的初始化使用括号的方式括号中是初始化的数值。这里三个变量分别是_name, _num, _grade, 然后通过形参name, num, grade对他们进行赋值。

初始化规则 

下面请仔细阅读。很复杂,很重要。

首先, 构造函数的执行方式顺序先初始化列表再函数体。

然后,初始化列表的执行顺序不是从上到下, 而是按照成员变量的声明顺序进行初始化。

初始化列表中没有初始化的对象, 比如_name, 就只能等到函数体再初始化。

初始化列表对内置类型就是如图中的初始化方式注意, 是内置类型才是上面的方式。自定义类型就没有这么简单了。

首先构造函数的执行顺序不会改变。 但是如果成员变量中有自定义类型的话, 它在初始化的时候, 会去调用自己相应的构造函数。 (因为对内置类型编译器可以直接进行赋值拷贝工作。但是对于自定义类型, 编译器无法自己进行复制拷贝工作。 所以只能去调用类型本身的构造函数。 这其实就是套娃的过程:如果调用的构造函数中还有自定义类型的初始化, 那么就要再去调用那个自定义类型的构造函数,直到调用的构造函数中没有自定义类型 

-------------------------------------------------------------------------------------------------------------------------

以上, 是将自定义类型显示在符号化列表初始化的情况。 如果没有在符号化列表初始化的话。 那么编译器会自动对该自定义类型变量进行初始化。并且初始化时不传参,就是这个意思

就相当于这里的_b进行初始化的时候没有传参。

以上, 就是初始化列表对于自定义类型以及内置类型初始化的规则。 

现在引入两个概念, 一个概念是:三种必须在初始化列表初始化的变量类型。 

第二个概念是默认构造函数的概念。 

三个必须初始化的变量类型

三个必须在初始化列表初始化的变量类型:引用类型const修饰的常变量, 以及上面讲的自定义类型。 

那么为什么会有这样的规定?

首先引用类型必须在定义的时候初始化类的实例化的同时也是对类的特征,属性进行初始化的时候。如果这一次没有初始化。 那么之后再对类的特征进行处理, 那不是初始化。 那叫赋值。 

所以必须对引用类型进行初始化。 

至于const修饰的常变量同样的道理, 因为必须在定义的时候初始化, 那么就必须在初始化列表初始化。 你可能有疑惑说为什么必须在初始化列表, 在构造函数的函数体初始化不行吗?其实, c++中规定了, 初始化列表的叫初始化, 函数体中的是赋值。 

-----------------------------------------------自定义类型很重要--------------------------------------------------

自定义类型为什么规定不做解释, 只需要知道这样规定就好。重点是自定义类型与上面两个有点差异。 虽然自定义类型也必须初始化。 但是如果我们不对它进行初始化, 编译器不会报错。 而是自己去调用自定义类型的默认构造函数。 这就叫自定义类型的隐式初始化

-------------------------------------------------------------------------------------------------------------------------

默认构造函数

那么什么是默认构造函数, 这就是我们的第二个要引入的概念:

默认构造函数就是:没有形参, 或者形参全缺省的构造函数。 

默认成员函数

在c++的类的成员函数中,有六个特殊的成员函数, 被称为默认成员函数

默认成员函数的意思就是如果不定义他们, 系统就会自动生成。

这里主要分析三个:默认构造函数

                                默认拷贝构造函数

                                析构函数

默认构造函数

首先, 什么是默认构造函数,这个概念在上面我们刚刚提到。 这里不做赘述。

那么编译器生成的默认构造函数什么样的?如图:

如图中红框框,编译器就会默认生程这样的默认构造函数。

由此我们可以知道。 这里生成的默认构造函数虽然是个空函数体, 空初始化列表。 但是通过上面的知识我们直到。 他会对自定义类型进行处理——自定义类型的隐式初始化。 

所以:默认构造函数的作用是对内置类型不做任何处理。 但是对于自定义类型会去调用它的默认构造函数

如果我们定义了任意一个构造函数, 编译器都不会生成默认构造函数。 

这个时候我们如果这样定义, 就会报错, 因为没有可以调用的构造函数。 

 但是我们如果不定义这个构造函数, 就可以编译通过, 因为此时编译器默认生成了一个空的默认构造函数, 不需要传参。 可以匹配p1

 有一个好办法就是定义全缺省的默认构造函数, 这样就可以一劳永逸, 不必担心传参是否有匹配的构造函数的问题:

默认拷贝构造函数

在学习默认拷贝构造函数之前我们要先学习拷贝构造函数

拷贝构造

拷贝构造是构造函数的一种重载形式。 它的参数是类本身实例化对象。意思就是将已经实例化的对象的值拷贝给将要实例化的对象。

 当我们使用对象初始化对象的时候就会调用拷贝构造。

默认拷贝构造

默认拷贝构造只对内置类型进行浅拷贝操作。如果是涉及到动态分配问题。那么就达不到我们的要求。 因为编译器不会对动态内存分配的内存里的数据进行拷贝。

像图中, 只是p1的_name成员保存的地址赋值给了p2的_name成员。

但是, 这不符合我们的要求。 所以我们就不能使用默认拷贝构造函数。 而是需要我们自己定义拷贝构造函数。 

 

像如图中自己定义的拷贝构造就是深拷贝。  

析构函数

拷贝构造一样, 在学习默认析构函数之前我们应该了解以下析构函数。

析构函数

析构函数就是对空间进行释放。 对于内置类型来说, 不需要定义析构函数, 因为编译器会自动对内置类型进行释放。 但是涉及到资源分配的问题。 编译器无法直接对资源分配空间进行释放。 所以就用到了析构函数。

析构函数的函数名称是类名前面加上~, 并且析构函数不接受任何传参。 同时析构函数最多只能有一个。

默认析构函数

编译器默认生成的析构函数就是默认构造函数。 默认构造函数的类体是空。对于内置类型不做处理, 对于自定义类型会去调用该自定义类型的构造函数。 

如果类中没有申请资源时,析构函数可以不写,直接使用编译器生成的默认析构函数即可;但是有资源申请时,否则会造成资源泄漏。

如何区分改写不该写呢?其实, 只要你自己定义了拷贝构造函数的时候, 就要自己手动定义析构函数。 因为只要你自己的定义了拷贝构造, 说明就涉及到了深拷贝问题。 深拷贝就是内存资源分配的问题。 就要定义析构。

以上就是本节的全部内容。

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

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

相关文章

【Java设计模式】九、桥接模式

文章目录 0、背景1、模式2、案例3、使用场景 0、背景 现要创建不同的图形,图形的形状有圆、长方形、正方形,且它们有不同的颜色 两个维度在变,使用类的继承可以实现,也符合开闭原则,但会类爆炸。 1、模式 将抽象与…

BUG:Enigma Virtual Box打包.net独立程序不正常

专栏介绍与文章目录-CSDN博客 尝试了使用Enigma Virtual Box打包.net8的winform程序,程序发布为不依赖框架的单一文件(设置了压缩),再和另外几个文件打包成一个exe(另外的文件不影响.net程序打开)。 但是打…

Vue中的组件:构建现代Web应用的基石

🤍 前端开发工程师、技术日更博主、已过CET6 🍨 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 🕠 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 🍚 蓝桥云课签约作者、上架课程《Vue.js 和 E…

动态规划:Leetcode 91.解码方法

题目 一条包含字母 A-Z 的消息通过以下映射进行了 编码 : A -> "1" B -> "2" ... Z -> "26" 要 解码 已编码的消息,所有数字必须基于上述映射的方法,反向映射回字母(可能有多种方法&am…

GNN-Transformer新突破!全局与局部的完美融合

图神经网络(GNN)和Transformer的结合是近年来的研究热点。这类结合不仅能够让两者发挥各自的优势,还能推动模型的创新,提高处理图数据的效率和性能。 具体点讲,通过利用Transformer,我们可以扩展GNN的感受…

Python办公自动化之PDF(二)

Python操作PDF二 1、PyMuPDF简介2、 1、PyMuPDF简介 PyMuPDF(也称Fitz)开源,提供了一整套用于处理PDF文件的综合工具。使用PyMuPDF,用户可以高效地执行打开PDF、提取文本、图像和表格、操作旋转和裁剪等页面属性、创建新PDF文档以…

Koltin 语言与Java语言有哪些差异?

目录 1. 变量声明方式不同 2. 方法(函数)定义略有不同 3. 逻辑控制关键字 if 的使用区别 4. 继承的区别 5. 接口和实现的区别 6. Koltin 的数据类有什么用? 1. 变量声明方式不同 Koltin声明变量的方式与Java类似,但是顺序不太一样。 举例如下 J…

Mapbox添加model图层

贴个群号 WebGIS学习交流群461555818,欢迎大家 效果图 源码 经常关注mapbox的朋友会发现,3.0版本之后,mapbox的api里更新了一个model图层 但是呢,却没有提供model有关的api,让我们摸不着头脑,到底该如…

20个常用的Python脚本

以下是20个常用的Python脚本示例&#xff1a; 计算阶乘 def factorial(n):if n 0:return 1else:return n * factorial(n-1)print(factorial(5))斐波那契数列 def fibonacci(n):if n < 1:return nelse:return fibonacci(n-1) fibonacci(n-2)print(fibonacci(10))判断素数…

最新基于R语言lavaan结构方程模型(SEM)技术

原文链接&#xff1a;最新基于R语言lavaan结构方程模型&#xff08;SEM&#xff09;技术https://mp.weixin.qq.com/s?__bizMzUzNTczMDMxMg&mid2247596681&idx4&sn08753dd4d3e7bc492d750c0f06bba1b2&chksmfa823b6ecdf5b278ca0b94213391b5a222d1776743609cd3d14…

【C++庖丁解牛】模拟实现STL的string容器(最后附源码)

&#x1f4d9; 作者简介 &#xff1a;RO-BERRY &#x1f4d7; 学习方向&#xff1a;致力于C、C、数据结构、TCP/IP、数据库等等一系列知识 &#x1f4d2; 日后方向 : 偏向于CPP开发以及大数据方向&#xff0c;欢迎各位关注&#xff0c;谢谢各位的支持 目录 1.vs和g下string结构…

librtmp交叉编译

目录 软件包下载 libopenssl编译 libz编译 librtmp编译 软件包下载 librtmp依赖openssl库和zlib库&#xff0c;注意openssl库版本需要为1.0&#xff0c;三个库下载地址如下&#xff1a; openssl:https://www.openssl.org/source/old/1.0.0/openssl-1.0.0s.tar.gz zlib:h…

LeetCode_Hot100_栈_155最小栈_Python

题目 设计一个支持 push &#xff0c;pop &#xff0c;top 操作&#xff0c;并能在常数时间内检索到最小元素的栈。 实现 MinStack 类: MinStack() 初始化堆栈对象。void push(int val) 将元素val推入堆栈。void pop() 删除堆栈顶部的元素。int top() 获取堆栈顶部的元素。i…

mysql笔记:3. 表数据更新

文章目录 插入数据插入单条数据插入多条数据复制表插入数据 更新数据删除数据截断表 存储在系统中的数据是数据库管理系统&#xff08;DBMS&#xff09;的核心&#xff0c;数据库被设计用来管理数据的存储、访问和维护数据的完整性。那么&#xff0c;数据库和表创建完成后&…

创造一款安卓自定义控件(4)——使用Matrix的setPolyToPoly方法实现图像纠正

接上文&#xff1a; 创造一款安卓自定义控件_任意4顶点裁剪框http://t.csdnimg.cn/vu1r5 创造一款安卓自定义控件_任意4顶点裁剪框2_为裁剪框添加放大镜功能http://t.csdnimg.cn/qkngh 创造一款安卓自定义控件_裁剪原理介绍http://t.csdnimg.cn/ORRRL 需求 随着需求修改&#x…

Stable Diffusion 详解

整体目标 文本生成图片&#xff1b;文本图片生成图片 网络结构 CLIP的文本编码器和图片生成器组成图像生成器&#xff0c;输入是噪声经过UNet得到图像特征&#xff0c;最后解码得到图像 前向扩散 模型直接预测图片难度比较大&#xff0c;所有让模型预测噪音然后输入-噪音…

macbook pro 2018 安装 arch linux 双系统

文章目录 友情提醒关于我的 mac在 mac 上需要提前做的事情复制 wifi 驱动 在 linux 上的操作还原 wifi 驱动连接 wifi 网络磁盘分区制作文件系统挂载分区 使用 archinstall 来安装 arch linux遗留问题 友情提醒 安装 archl linux 的时候&#xff0c;mac 的键盘是没法用的&#…

设计模式-结构型模式-代理模式

代理模式&#xff08;Proxy&#xff09;&#xff0c;为其他对象提供一种代理以控制对这个对象的访问。[DP] // 定义接口 interface Subject {void request(); }// 真实主题对象 class RealSubject implements Subject {Overridepublic void request() {System.out.println(&quo…

网络基础『 序列化与反序列化』

&#x1f52d;个人主页&#xff1a; 北 海 &#x1f6dc;所属专栏&#xff1a; Linux学习之旅、神奇的网络世界 &#x1f4bb;操作环境&#xff1a; CentOS 7.6 阿里云远程服务器 文章目录 &#x1f324;️前言&#x1f326;️正文1.协议的重要性2.什么是序列化与反序列化&…

AI会砸了我们的饭碗?

Sora&#xff0c;由OpenAI推出&#xff0c;是一款创新的文本到视频生成模型。它能够将文本描述转化为引人入胜的高清视频片段。采用了扩散模型和变换器架构&#xff0c;Sora实现了高效的训练。其方法包括统一表示法、基于补丁的表示法、视频压缩网络和扩散变换器。 Sora具备多种…