【C++入门】拷贝构造运算符重载

目录

 1. 拷贝构造函数

 1.1 概念

 1.2 特征

 1.3 常用场景

 2. 赋值运算符重载

2.1 运算符重载

 2.2 特征

 2.3 赋值运算符


前言

        拷贝构造和运算符重载是面向对象编程中至关重要的部分,它们C++编程中的一个核心领域,本期我详细的介绍拷贝构造和运算符重载。

在这里插入图片描述

 1. 拷贝构造函数

    为什么要有拷贝构造?

        首先肯定是有需求才会产生,在C语言中,使用内置类型时,时常会有这样的场景:使用一个已经存在的变量赋值给新变量(也就是一种拷贝),把一个变量的值拷贝给另一个变量。那么在C++中我们也会遇到,在实例化对象时也需要将一个对象的数据拷贝给另一个对象的场景,拷贝构造函数是为了方便自定义类型的数据拷贝而设计的。

 1.1 概念

        拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),用已存在的类类型对象创建新对象时由编译器自动调用

 1.2 特征

        拷贝构造函数也是特殊的成员函数

 特征:

  • 拷贝构造函数是构造函数的一个重载形式
  • 拷贝构造函数的参数只有一个且必须是类类型对象的引用

 使用传值方式编译器直接报错,因为会引发无穷递归调用

class Date
{
public:Date(int year = 1900, int month = 1, int day = 1){_year = year;_month = month;_day = day;}//Date(const Date d)// 错误写法:编译报错,会引发无穷递归Date(const Date& d){_year = d._year;_month = d._month;_day = d._day;}
private:int _year;int _month;int _day;
};
int main()
{Date d1;Date d2(d1);return 0;
}

        在传值调用中,形参是实参的拷贝,在d1传给函数里的d时会调用拷贝构造(把d1拷贝给d,Date d(d1)),d1拷贝给调用函数体的d继续调用拷贝构造……这样就形成了无限递归。

  •  若未显式定义,编译器会生成默认的拷贝构造函数

        默认的拷贝构造函数对象按内存存储按字节序完成拷贝,这种拷贝叫做浅拷贝,或者值拷贝。浅拷贝在一些场景中并不适用,比如:

自定义类型:栈

class Stack
{
public:Stack(size_t capacity = 3){cout << "Stack(size_t capacity = 3)" << endl;_a = (int*)malloc(sizeof(int) * capacity);if (nullptr == _a){perror("malloc申请空间失败!!!");}_capacity = capacity;_top = 0;}~Stack(){cout << "~Stack()" << endl;free(_a);_capacity = _top = 0;_a = nullptr;}private:int* _a;int _capacity;int _top;
};
void fun1(Stack st)
{//…
}
int main()
{Stack st1;fun(st1);return 0;
}

这个代码在运行时就会报错。

         函数调用时是传值调用,形参是实参的拷贝,所以fun函数里的st和main函数里的st1指定的是同一块空间

 在fun执行结束时,st就会销毁,调用析构会将st中的_a指向的空间销毁

 main函数结束时,st1也会调用析构函数销毁_a指向的空间

 但是这块空间已经被销毁,再去销毁就会造成内存重复释放。

 所以必要时我们还需要进行深拷贝(自己实现)

Stack(const Stack& stt){cout << "Stack(Stack& stt)" << endl;// 深拷贝_a = (int*)malloc(sizeof(int) * stt._capacity);if (_a == nullptr){perror("malloc fail");exit(-1);}memcpy(_a, stt._a, sizeof(int) * stt._top);_top = stt._top;_capacity = stt._capacity;}

        深拷贝是重新开一块空间存放实参中_a指向空间的数据,在销毁时也是释放各自的空间,不会造成同块空间被重复释放的情况。

 类中如果没有涉及资源申请时,拷贝构造函数是否写都可以;

一旦涉及到资源申请时,则拷贝构造函数是一定要写的,否则就是浅拷贝

 注意:在编译器生成的默认拷贝构造函数中,内置类型是按照字节方式直接拷贝的,而自定
 义类型调用其拷贝构造函数完成拷贝

 1.3 常用场景

  • 使用已存在对象创建新对象
int main()
{Date d1(2023,11,11);Date d2(d1);return 0;
}
  • 函数返回值类型为类类型对象
Date Test(Date d)
{
Date tmp(d);
return tmp;//返回时有一次拷贝
}
  • 函数参数类型为类类型对象
void fun1(Stack st)//传值时有一次拷贝
{//…
}

为了提高程序效率,一般对象传参时,尽量使用引用类型,返回时根据实际场景,能用引用
尽量使用引用。

 2. 赋值运算符重载

2.1 运算符重载

     C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似

 2.2 特征

 函数名字为:关键字operator后面接需要重载的运算符符号

 函数原型:返回值类型 operator操作符(参数列表)

  • 不能通过连接其他符号来创建新的操作符:比如operator@
  • 重载操作符必须有一个类类型参数
  • 用于内置类型的运算符,其含义不能改变,例如:内置的整型+,不 能改变其含义
  • 作为类成员函数重载时,其形参看起来比操作数数目少1,因为成员函数的第一个参数为隐藏的this
  • .* 、::(域作用限定符) 、sizeof 、?  :(三目运算符) 、‘ .(点)’、 注意以上5个运算符不能重载。这个经常在笔试选择题中出现

 2.3 赋值运算符

  介绍完基本特征,我们来试着用一下,以日期类为例:

class Date
{private:int _year;int _month;int _day;
};

        日期类的内部数据类型都是内置类型,这里的析构函数、构造函数、拷贝构造都可以不写(编译器默认生成),我们在写运算符重载时是在类的里边还是外边?

        其实都是可以的,但为了保证封装性,最好还是将运算符重载写到类里,(要想访问需要把类的成员变量置为public,正常情况下,类的成员变量都是私有的,在类外部无法访问)并且在调用时也很不方便。

 以赋值运算符为例:

class Date
{
public:Date& operator=(const Date& d)
{
if(this != &d){_year = d._year;_month = d._month;_day = d._day;}return *this;
}private:int _year;int _month;int _day;
};

 赋值运算符只能重载成类的成员函数不能重载成全局函数

 赋值运算符重载成全局函数,注意重载成全局函数时没有this指针了,需要给两个参数

class Date
{
public:int _year;int _month;int _day;
};Date& operator=(Date& left, const Date& right)
{if (&left != &right){left._year = right._year;left._month = right._month;left._day = right._day;}return left;
}
// 编译失败:
// error C2801: “operator =”必须是非静态成员

原因:赋值运算符如果不显式实现,编译器会生成一个默认的。此时用户再在类外自己实现
一个全局的赋值运算符重载,就和编译器在类中生成的默认赋值运算符重载冲突。

 用户没有显式实现时,编译器会生成一个默认赋值运算符重载

 同样,编译器默认生成的赋值运算符,以值的方式逐字节拷贝

 注意:

内置类型成员变量是直接赋值的,而自定义类型成员变量需要调用对应类的赋值运算符
重载完成赋值

 但是编译器的默认生成的并不能满足某些场景(如栈):

 在栈的赋值运算符重载时需要进行深拷贝,如果不深拷贝就会造成与拷贝构造一样的结果,空间被重复释放。

class Stack
{
public:Stack(size_t capacity = 6){cout << "Stack(size_t capacity = 3)" << endl;_a = (int*)malloc(sizeof(int) * capacity);if (nullptr == _a){perror("malloc申请空间失败!!!");}_capacity = capacity;_top = 0;}void Push(const int& data){// CheckCapacity();_a[_top] = data;_top++;}Stack(const Stack& stt){cout << "	Stack(Stack& stt)" << endl;// 深拷贝_a = (int*)malloc(sizeof(int) * stt._capacity);if (_a == nullptr){perror("malloc fail");exit(-1);}memcpy(_a, stt._a, sizeof(int) * stt._top);_top = stt._top;_capacity = stt._capacity;}~Stack(){cout << "~Stack()" << endl;free(_a);_capacity = _top = 0;_a = nullptr;}Stack& operator=(const Stack& s){cout << "operator=(const Stack& s)" << endl;if (this != &s) // 检查自我赋值{free(_a);//清理原有数据// 深拷贝_a = (int*)malloc(sizeof(int) * s._capacity);if (_a == nullptr){perror("malloc fail");exit(-1);}memcpy(_a, s._a, sizeof(int) * s._top);_top = s._top;_capacity = s._capacity;}return *this;}private:int* _a;int _capacity;int _top;
};

注意:如果类中未涉及到资源管理,赋值运算符是否实现都可以;一旦涉及到资源管理则必
须要实现


总结

         拷贝构造函数和运算符重载是 C++ 中重要的语言特性,掌握它们对于理解和设计复杂的程序非常重要。好了以上便是本期全部内容,最后,感谢阅读!

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

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

相关文章

Transformer中WordPiece/BPE等不同编码方式详解以及优缺点

❤️觉得内容不错的话&#xff0c;欢迎点赞收藏加关注&#x1f60a;&#x1f60a;&#x1f60a;&#xff0c;后续会继续输入更多优质内容❤️ &#x1f449;有问题欢迎大家加关注私戳或者评论&#xff08;包括但不限于NLP算法相关&#xff0c;linux学习相关&#xff0c;读研读博…

FPGA设计时序约束八、others类约束之Set_Case_Analysis

目录 一、序言 二、Set Case Analysis 2.1 基本概念 2.2 设置界面 2.3 命令语法 2.4 命令示例 三、工程示例 四、参考资料 一、序言 在Vivado的时序约束窗口中&#xff0c;存在一类特殊的约束&#xff0c;划分在others目录下&#xff0c;可用于设置忽略或修改默认的时序…

uniapp 微信小程序登录 新手专用 引入即可

预览 第一步导入插件 在引入的页面的登录按钮下拷贝一下代码 <template><view class"content"><button type"primary" click"login">微信登录</button></view><TC-WXlogin :wxloginwxlogin /> </templ…

05-Spring Boot工程中简化开发的方式Lombok和dev-tools

简化开发的方式Lombok和dev-tools Lombok常用注解 Lombok用标签方式代替构造器、getter/setter、toString()等重复代码, 在程序编译的时候自动生成这些代码 注解名功能NoArgsConstructor生成无参构造方法AllArgsConstructor生产含所有属性的有参构造方法,如果不希望含所有属…

[C/C++]数据结构 栈和队列()

一:栈 1.1 栈的概念及结构 栈是一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作,进行数据插入和删除操作的一端称为栈顶,另一端称为栈底,栈中的数据元素遵守先进后出的原则. 压栈:栈的插入操作叫做进栈/压栈/入栈,将数据插入栈顶 出栈:栈的删除操作也叫出…

Redis新操作

1.Bitmaps 1.1概述 Bitmaps可以对位进行操作&#xff0c;实际上它就是一个字符串&#xff0c;可以将Bitmaps想象为一个以位为单位的数组&#xff0c;数组中的每个元素只能存储0或者1&#xff0c;数组的下标在Bitmaps被称为偏移量。 setbit key offset value&#xff1a;设置o…

MIB 6.S081 System calls(1)using gdb

难度:easy In many cases, print statements will be sufficient to debug your kernel, but sometimes being able to single step through some assembly code or inspecting the variables on the stack is helpful. To learn more about how to run GDB and the common iss…

十二.Jenkins持续集成

十二.Jenkins持续集成 一.安装jenkins 1.下载 Jenkins下载地址&#xff1a;http://jenkins-ci.org/ 或 https://mirrors.jenkins-ci.org/redhat/2.安装 可以通过官网的安装方式来安装 安装完后&#xff0c;需要修改以下的配置 vim /usr/lib/systemd/system/jenkins.servic…

git常用命令和参数有哪些?【git看这一篇就够了】

文章目录 前言常用命令有哪些git速查表奉上常用参数后言 前言 hello world欢迎来到前端的新世界 &#x1f61c;当前文章系列专栏&#xff1a;git操作相关 &#x1f431;‍&#x1f453;博主在前端领域还有很多知识和技术需要掌握&#xff0c;正在不断努力填补技术短板。(如果出…

Git企业开发级讲解(五)

&#x1f4d8;北尘_&#xff1a;个人主页 &#x1f30e;个人专栏:《Linux操作系统》《经典算法试题 》《C》 《数据结构与算法》 ☀️走在路上&#xff0c;不忘来时的初心 文章目录 一、bug 分⽀二、删除临时分支三、小结 一、bug 分⽀ 假如我们现在正在 dev2 分⽀上进⾏开发…

AI Navigation导航系统_unity基础开发教程

AI Navigation导航系统 安装插件烘焙导航系统障碍物创建人物的AI导航动态障碍物 在unity编辑器中&#xff0c;有一个灰常好用的插件&#xff1a;Navigation。有了它1&#xff0c;你就可以实现人物自动走到你鼠标点击的位置&#xff0c;而且还会自动避开障碍物&#xff0c;下面就…

第7天:信息打点-资产泄漏amp;CMS识别amp;Git监控amp;SVNamp;DS_Storeamp;备份

第7天&#xff1a;信息打点-资产泄漏&CMS识别&Git监控&SVN&DS_Store&备份 知识点&#xff1a; 一、cms指纹识别获取方式 网上开源的程序&#xff0c;得到名字就可以搜索直接获取到源码。 cms在线识别&#xff1a; CMS识别&#xff1a;https://www.yun…

quickapp_快应用_tabBar

tabBar 配置项中配置tabBar(版本兼容)使用tabs组件配置tabBar语法示例问题-切换tab没有反应问题-数据渲染问题解决优化 问题-tab的动态配置 第三方组件tabbar 一般首页都会显示几个tab用于进行页面切换&#xff0c;以下是几种tab配置方式。 配置项中配置tabBar(版本兼容) 在m…

系列五、怎么查看默认的垃圾收集器是哪个?

一、怎么查看默认的垃圾收集器是哪个 java -XX:PrintCommandLineFlags -version

wpf devexpress 创建布局

模板解决方案 例子是一个演示连接数据库连接程序。打开RegistrationForm.BaseProject项目和如下步骤 RegistrationForm.Lesson1 项目包含结果 审查Form设计 使用LayoutControl套件创建混合控件和布局 LayoutControl套件包含三个主控件&#xff1a; LayoutControl - 根布局…

【Go入门】 Go搭建一个Web服务器

【Go入门】 Go搭建一个Web服务器 前面小节已经介绍了Web是基于http协议的一个服务&#xff0c;Go语言里面提供了一个完善的net/http包&#xff0c;通过http包可以很方便的搭建起来一个可以运行的Web服务。同时使用这个包能很简单地对Web的路由&#xff0c;静态文件&#xff0c…

【力扣】从零开始的动态规划

【力扣】从零开始的动态规划 文章目录 【力扣】从零开始的动态规划开头139. 单词拆分解题思路 45. 跳跃游戏 II解题思路 5. 最长回文子串解题思路 1143. 最长公共子序列解题思路 931. 下降路径最小和解题思路 开头 本力扣题解用5题来引出动态规划的解题步骤&#xff0c;用于本…

Android图片涂鸦,Kotlin(1)

Android图片涂鸦&#xff0c;Kotlin&#xff08;1&#xff09; import android.content.Context import android.graphics.Canvas import android.graphics.Color import android.graphics.Paint import android.graphics.Path import android.graphics.PointF import android.…

UE4动作游戏实例RPG Action解析三:实现效果,三连击Combo,射线检测,显示血条,火球术

一、三连Combo 实现武器三连击,要求: 1.下一段Combo可以随机选择, 2.在一定的时机才能再次检测输入 3. 等当前片段播放完才播放下一片段 1.1、蒙太奇设置 通过右键-新建蒙太奇片段,在蒙太奇里创建三个片段,并且移除相关连接,这样默认只会播放第一个片段 不同片段播…

图像分类(六) 全面解读复现MobileNetV1-V3

MobileNetV1 前言 MobileNetV1网络是谷歌团队在2017年提出的&#xff0c;专注于移动端和嵌入设备的轻量级CNN网络&#xff0c;相比于传统的神经网络&#xff0c;在准确率小幅度降低的前提下大大减少模型的参数与运算量。相比于VGG16准确率减少0.9%&#xff0c;但模型的参数只…