『C++成长记』拷贝构造函数

news2025/1/22 17:03:23

 🔥博客主页:小王又困了

📚系列专栏:C++

🌟人之为学,不日近则日退

❤️感谢大家点赞👍收藏⭐评论✍️

目录

一、拷贝构造函数

📒1.1拷贝构造函数的概念

📒1.2拷贝构造函数的写法

📒1.3编译器生成的拷贝构造

📒1.4拷贝构造函数的用法

📒1.5拷贝构造函数典型调用场景


一、拷贝构造函数

📒1.1拷贝构造函数的概念

     拷贝构造函数,是一种特殊的构造函数。这种构造函数由编译器用,用于基于同一类的其他对象的构建及初始化,也就是是创建对象的时候,用一个已存在的对象,去初始化待创建的对象。拷贝构造函数的形参必须是引用,但并不限制为const,一般普遍的会加上const限制。

例如,如果我们有一个名为Date的类,并且我们希望创建一个新的Date对象作为现有Date对象的副本,我们可以使用拷贝构造函数来实现这一点。在这种情况下,新的Date对象将具有与现有对象相同的属性值

class Date
{
private:
    // 基本类型(内置类型)
    int _year = 1970;
    int _month = 1;
    int _day = 1;
};

Data d1;//定义一个日期类对象d1
Data d2(d1);//会去调用拷贝构造函数

拷贝构造函数是针对自定义类型的,自定义类型的对象在拷贝的时候,C++规定必须要调用拷贝构造函数。 

📒1.2拷贝构造函数的写法

在上面的概念中提到,拷贝构造函数的形参必须是引用,这是为什么呢?

🎀形参不是引用的情况:

Data(Data d)//错误的拷贝构造
{
    _year = d._year;
    _month = d._month;
    _day = d._day;
}

Test()
{
    Date d1(2023, 11, 20);
    Date d2(d1);
}

     我们创建一个d2对象,把实参d1传递过去,然后用形参 接收,最后把形参 的值赋给this指针(this指针指向的是d2),到这里我们觉得一切正常,但是这段代码有很大的错误。

🗒️为什么错误

问题就在于,我们传参时没有使用引用,上面代码传参使用的是值传递,值传递形参是实参的一份临时拷贝,拷贝也就是要形参d实参d1有相同的属性值,所以还要调用拷贝构造,这里就会引发无穷递归调用。

 形参 在接收实参d1的时候,又要去调用拷贝构造来创建 ,这次调用拷贝构造,又会有一个形参 ,这个形参又需要调用拷贝构造才能创建,一直递归调用。为了避免出现这种无穷递归,编译器会自行检查,如果拷贝构造函数的形参是值传递,编译时会直接报错。

🎀形参是引用的情况:

     为了使代码不在无穷递归,拷贝构造函数的形参只能有一个,并且必须是类类型对象的引用。下面才是正确的拷贝构造函数:

Data(Data& d)//正确的拷贝构造
{
    _year = d._year;
    _month = d._month;
    _day = d._day;
}
Data d1(2023, 7, 20);//定义一个日期类对象d1
Data d2(d1);

这里形参 d1的别名,它两共用一块空间,此时就不会再去无穷无尽的调用拷贝构造。

📒1.3编译器生成的拷贝构造

     拷贝构造是一种默认成员函数,我们不写编译器会自动生成。默认的拷贝构造函数对内置类型按内存存储按字节序完成拷贝,这种拷贝叫做浅拷贝,或者值拷贝;对自定义类型是调用其拷贝构造函数完成拷贝。

class Time//定义时间类
{
public:
    Time()//普通构造函数
    {
        _hour = 1;
        _minute = 1;
        _second = 1;
    }
    Time(const Time& t)//拷贝构造函数
    {
        _hour = t._hour;
        _minute = t._minute;
        _second = t._second;
        cout << "Time::Time(const Time&)" << endl;
    }
private://成员变量
    int _hour;
    int _minute;
    int _second;
};

class Date
{
private:
	// 基本类型(内置类型)
	int _year = 1970;
	int _month = 1;
	int _day = 1;
	// 自定义类型
	Time _t;
};

int main()
{
    Date d1;
    // 用已经存在的d1拷贝构造d2,此处会调用Date类的拷贝构造函数
    // 但Date类并没有显式定义拷贝构造函数,则编译器会给Date类生成一个默认的拷贝构造函数
    Date d2(d1);
    return 0;
}

📒1.4拷贝构造函数的用法

     编译器默认生成的构造函数对内置类型和自定义类型都做了处理。那我们是不是就可以不写拷贝构造函数了呢?答案是否定的,对于日期类,我们可以不写,用编译器自己生成的,但是对于一些需要自己开辟空间对象,要进行深拷贝,构造函数是非写不可的。栈就是一个典型的需要我们自己写构造函数的例子:

typedef int DataType;
class Stack
{
public:
    Stack(size_t capacity = 10)
    {
        _array = (DataType*)malloc(capacity * sizeof(DataType));
        if (nullptr == _array)
        {
            perror("malloc申请空间失败");
            return;
	    }
        _size = 0;
        _capacity = capacity;
    }
    void Push(const DataType& data)
    {
        // CheckCapacity();
        _array[_size] = data;
        _size++;
    }
    ~Stack()
    {
        if (_array)
        {
            free(_array);
            _array = nullptr;
            _capacity = 0;
            _size = 0;
        }
    }
private:
    DataType *_array;
    size_t _size;
    size_t _capacity;
};
int main()
{
    Stack s1;
    s1.Push(1);
    s1.Push(2);
    Stack s2(s1);
    return 0;
}

如上代码,定义了一个Stack类,栈中的成员变量都是内置类型。我们没有写它的拷贝构造函数,编译器默认生成的拷贝构造函数会对这三个成员变量都完成值拷贝(浅拷贝)。

🗒️浅拷贝:

     浅拷贝是创建一个新的对象,并把原有的对象属性值完整地拷贝过来。这种拷贝方式既包括了原始类型的值,也包括了引用类型的内存地址。对于数据类型是基本数据类型的成员变量,浅拷贝会直接进行值传递,也就是将该属性值复制一份给新的对象。

     s2调用拷贝构造函数,Stack类中没有显示定义的拷贝构造函数,则调用编译器默认生成的拷贝构造函数,将s1的值拷贝到s2中,因此s1s2指向同一块内存空间。当程序退出,s1s2要销毁,s2先销毁,s2销毁时调用析构函数,已经将0x11223344这块空间释放了,但是s1并不知道,到s1销毁的时候,会将0x11223344这块空间再释放一次,一块内存空间多次释放,最终就会导致程序崩溃。

🗒️深拷贝:

     通过上面的分析可以看出,简单的浅拷贝不能满足栈的需求,因此,对于栈,我们需要自己写一个拷贝构造函数,来实现深拷贝。深拷贝是对对象具体内容进行复制,它会创建一个新的对象实例,并复制所有属性以及这些属性指向的动态分配的内存

//自己写的拷贝构造函数,实现深拷贝
Stack(const Stack& st)
{
    DataType* tmp = (DataType*)malloc(sizeof(DataType) * st._capacity);
    if (nullptr == tmp)
    {
        perror("malloc申请空间失败");
        return;
    }
    memcpy(tmp, st._array, sizeof(DataType) * st._size);
    _array = tmp;
    _size = st._size;
    _capacity = st._capacity;
}

深浅拷贝的区别在于他们处理对象内存的方式不同。浅拷贝新旧对象还是共享同一块内存,改变其中一个,另一个也会受影响。而深拷贝则会复制出一个全新的对象实例,新对象跟原对象不共享内存,两者操作互不影响。因此在某些情况下,浅拷贝可能会使旧对象和新对象产生相互影响,这可能会导致数据的不一致。在这种情况下,你可能需要自定义深拷贝构造函数来创建一个新的、独立的对象实例。

注意:类中如果没有涉及资源申请时,拷贝构造函数写不写都可以;一旦涉及到资源申请时,拷贝构造函数是一定要写的,否则就是浅拷贝。

📒1.5拷贝构造函数典型调用场景

  • 使用已存在对象创建新对象
  • 函数参数类型为类类型对象
  • 函数返回值类型为类类型对象
class Data
{
public:
    Data(int year = 1, int month = 1, int day = 1)
    {
        cout << "调用构造函数:" << this << endl;
        cout << endl;
        _year = year;
        _month = month;
        _day = day;
    }

    Data(const Data& d)
    {
        cout << "调用拷贝构造:" << this << endl;
        cout << endl;
        _year = d._year;
        _month = d._month;
        _day = d._day;
    }

    ~Data()
    {
        cout << "~Data()" << this << endl;
        cout << endl;
    }
private:
    int _year;
    int _month;
    int _day;

    //可以不用写析构,因为全是自定义类型,并且没有动态申请的空间,这三个成员变量会随着对象生命周期的结束而自动销毁
};
Data Text(Data x)
{
    Data tmp;
    return tmp;
}

int main()
{
	Data d1(2023, 4, 29);
	Text(d1);
	return 0;
}

 📖总结:
    自定义类型在传参的时候,形参最好用引用来接收,这样可以避免调用拷贝构造函数,尤其是深拷贝的时候,会大大的提高效率,函数返回时,如果返回的对象在函数栈帧销毁后还在,最好也用引用返回。


🎁结语: 

     本次的内容到这里就结束啦。希望大家阅读完可以有所收获,同时也感谢各位读者三连支持。文章有问题可以在评论区留言,博主一定认真认真修改,以后写出更好的文章。你们的支持就是博主最大的动力。

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

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

相关文章

SAP 散装物料简介

散装物料(Bulk Material),也叫做间接物料(Indirect Material),是一般企业在库存管理时常见的一种物料形式。散装物料专指那些价值小、消耗量大、消耗率高的物料件。这些物料组件同样服务于企业的生产活动,并且在企业的工作中心中被生产活动直接消耗(如螺丝钉、润滑油、…

使用飞书自定义机器人发送消息

使用飞书机器人可以很方便的获取自动化任务的反馈&#xff1a; 在群里创建一个机器人&#xff1a; 记住下面的 webhook地址&#xff0c;这个是标识机器人的唯一ID&#xff0c;比如它的webhook地址是&#xff1a;"https://open.feishu.cn/open-apis/bot/v2/hook/xxxxxxx-a…

方案分享:如何做好云中的DDoS防御?

所有企业都会有遭受DDoS攻击的风险。由于目前DDoS即服务&#xff08;DaaS&#xff09;的售价低廉&#xff0c;因此对于恶意攻击者来说&#xff0c;发起攻击比以往任何时候都更加容易&#xff0c;技术门槛也更低。分析公司IDC一项关于DDoS防御的调查显示&#xff0c;超过50%的IT…

python自动化测试实战 —— 自动化测试框架的实例

软件测试专栏 感兴趣可看&#xff1a;软件测试专栏 自动化测试学习部分源码 python自动化测试相关知识&#xff1a; 【如何学习Python自动化测试】—— 自动化测试环境搭建 【如何学习python自动化测试】—— 浏览器驱动的安装 以及 如何更…

混频原理与频谱搬移

文章目录 混频器频谱搬移何为镜像频率信号的复混频参考 混频器 混频器是一个三端器件&#xff0c;两个输入一个输出&#xff0c;输出信号等于输入信号的乘积。所以混频器可以将两个不同频率的信号通过相乘的方式&#xff0c;从而产生原本两个频率和与差的新信号。 数学推导举…

vue文件下载请求blob文件流token失效的问题

页面停留很久token失效没有刷新页面&#xff0c;这时候点击下载依然可以导出文件&#xff0c;但是文件打不开且接口实际上返回的是401&#xff0c;这是因为文件下载的方式通过window创建a标签的形式打开的&#xff0c;并没有判断token失效问题 const res await this.$axios.…

Unity | Shader基础知识(第一集:unity中最简单的shader)

一、unity的shader unity写的shader并不是真正意义上的shader。 简单解释&#xff1a;真正的shader语言写起来还是麻烦的&#xff0c;unity希望大家写起来简单一点&#xff0c;于是在原来的基础上&#xff0c;给大家优化了一个语言&#xff0c;叫shaderLab&#xff0c;所以我…

PyTorch深度学习实战(25)——自编码器

PyTorch深度学习实战&#xff08;25&#xff09;——自编码器 0. 前言1. 自编码器2. 使用 PyTorch 实现自编码器小结系列链接 0. 前言 自编码器 (Autoencoder) 是一种无监督学习的神经网络模型&#xff0c;用于数据的特征提取和降维&#xff0c;它由一个编码器 (Encoder) 和一…

智慧燃气让城市能源系统高效运行

关键词&#xff1a;智慧燃气、燃气数字化、智慧燃气平台、智慧燃气解决方案、智慧燃气系统 随着我国城镇燃气行业的发展&#xff0c;燃气行业管理及服务从简单的手工运作阶段迈入数字燃气阶段&#xff0c;大量采用信息化手段管理燃气业务&#xff0c;智慧燃气应运而生。它既是…

互动时代的新趋势

数字人直播是指通过数字化技术将虚拟人物以真实的方式呈现在观众面前&#xff0c;并实现与观众的实时互动的一种新型娱乐形式。近年来&#xff0c;随着科技的发展和社交媒体的兴起&#xff0c;数字人直播越来越受到人们的关注&#xff0c;并成为当今互动时代的新趋势。 首先&a…

centos7x 安装支持gpu驱动的docker

1、卸载以前版本的驱动 sudo /usr/bin/nvidia-uninstall2、先安装基础项 yum install kernel kernel-devel gcc make -yyum install kernel kernel-devel gcc gcc-c make -y 3、禁用驱动源 nouveau echo "blacklist nouveau " >>/etc/modp…

移动CRM:智能化助力,销售更高效

商机稍纵即逝&#xff0c;及时把握机会才能促成更多交易。想要在出差途中也能掌握公司运营情况&#xff1f;得益于移动互联网的快速发展&#xff0c;现在您可以利用移动CRM客户管理系统&#xff0c;实时管理线索&#xff0c;查看销售数据。本文将向您介绍&#xff0c;移动CRM能…

热分析和报告软件 :FLIR Thermal Studio Pro Crack

FLIR 热工作室套件。FLIR Thermal Studio Pro FLIR Thermal Studio Suite 可帮助用户简化检查、组织数据并管理数千张热图像和视频——无论他们使用的是手持式热像仪、无人机系统 (UAS)、声学成像相机还是光学气体成像&#xff08;OGI&#xff09;相机。该订阅软件提供了关键组…

Qt提升绘制效率,绘制加速。

在我们绘制一些复杂逻辑且数据量巨大的图形时&#xff0c;经常会出现流畅性问题&#xff0c;本文就是来进行讲解如何提升绘制效率的。 实现思路&#xff1a; 场景1&#xff1a;我们绘制多个静态图形和绘制一张图片哪个更快。很明显绘制多个图形的时候要慢很多。所以我们将多个图…

Web漏洞分析-文件解析及上传(上)

随着互联网的迅速发展&#xff0c;网络安全问题变得日益复杂&#xff0c;而文件解析及上传漏洞成为攻击者们频繁攻击的热点之一。本文将深入研究文件解析及上传漏洞&#xff0c;通过对文件上传、Web容器IIS、命令执行、Nginx文件解析漏洞以及公猫任意文件上传等方面的细致分析&…

* demo、源码、桌面端软件

demo demohttps://bidding-m.gitee.io/maptalks-test-next/#/ 源码 源码https://gitee.com/bidding-M/maptalks-test-next 桌面端 桌面端https://gitee.com/bidding-M/map-collection/blob/master/apps/maptalks-win-0.0.1-x64.exe

中洺科技-数据标注创就业浪潮

平凡的事不能普通的做那么就有所不同。这也是我们在这波新的AI浪潮中做副业、创业时应该思考的。 正如一些业内从业者所说&#xff0c;数据标注行业本身的利润其实非常有限。如果现在招聘专职数据标注人员的成本太高&#xff0c;只做标注项目的公司就太难立足了&#xff0c;数据…

微信小程序(一) —— 常见组件

文章目录 &#x1f380;项目基本组成结构&#x1f4e2;常见的视图容器类组件viewscroll-viewswiper和swiper-item使用viewscroll-viewswiper和swiper-itemswiper标签属性 &#x1f336;️常用的基础内容组件textrich-text &#x1f4ee;其他常用组件buttonimagenavigator &…

简易的JS逆向解码

在实战的漏洞挖掘中阅读JS有以下几个作用&#xff1a; 1.JS中存在插件名字&#xff0c;根据插件找到相应的漏洞直接使用 通过控制台大致阅读网站JS代码发现此网页引用了北京的一家公司的代码&#xff0c;并且使用了h-net的框架&#xff0c;接下来我们可以百度这家公司或者是这…

基于C/C++的rapidxml加载xml大文件 - 上部分翻译

RAPIDXML手册 版本 1.13 版权所有 &#xff08;C&#xff09; 2006&#xff0c; 2009 Marcin Kalicinski有关许可证信息&#xff0c;请参阅随附的文件许可证 .txt。 目录 1. 什么是 RapidXml&#xff1f; 1.1 依赖性和兼容性1.2 字符类型和编码1.3 错误处理1.4 内存分配1.5 …