前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >C++进阶之路:何为拷贝构造函数,深入理解浅拷贝与深拷贝(类与对象_中篇)

C++进阶之路:何为拷贝构造函数,深入理解浅拷贝与深拷贝(类与对象_中篇)

作者头像
Srlua
发布于 2024-05-26 01:38:30
发布于 2024-05-26 01:38:30
50500
代码可运行
举报
文章被收录于专栏:CSDN社区搬运CSDN社区搬运
运行总次数:0
代码可运行

拷贝构造函数

概念 :

在创建对象时,可否创建一个与已存在对象一某一样的新对象呢?

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

特征:

拷贝构造函数也是特殊的成员函数,其特征如下:

1. 拷贝构造函数是构造函数的一个重载形式
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <iostream>

class MyClass {
private:
    int* data;

public:
    // 默认构造函数
    MyClass() {
        data = new int(0);
    }

    // 拷贝构造函数
    MyClass(const MyClass& other) {
        data = new int(*other.data);
    }

    // 析构函数
    ~MyClass() {
        delete data;
    }

    // 设置数据
    void setData(int value) {
        *data = value;
    }

    // 获取数据
    int getData() const {
        return *data;
    }
};

int main() {
    MyClass obj1;
    obj1.setData(42);

    // 使用拷贝构造函数创建新对象
    MyClass obj2(obj1);

    std::cout << "obj1 data: " << obj1.getData() << std::endl;
    std::cout << "obj2 data: " << obj2.getData() << std::endl;

    return 0;
}

在上述示例中,MyClass 类拥有一个指针 data,在默认构造函数中为其分配内存,并在析构函数中释放内存。拷贝构造函数通过使用 new 运算符,在堆上分配新的内存,并将原对象的数据复制到新内存中。

运行示例代码,输出结果为:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
obj1 data: 42
obj2 data: 42

可以看到,通过拷贝构造函数创建的新对象 obj2 具有与原对象 obj1 相同的数据。


2. 拷贝构造函数的参数只有一个必须是类类型对象的引用

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

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
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;
}
3.若未显式定义,编译器会生成默认的拷贝构造函数。

默认的拷贝构造函数对象按内存存储按字节序完成拷贝,这种拷贝叫做浅拷贝,或者值拷贝。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
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;
}

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

4. 编译器生成的默认拷贝构造函数已经可以完成字节序的值拷贝了,还需要自己显式实现吗?

如果一个类没有指针或引用等需要特别注意的成员变量,那么编译器生成的默认拷贝构造函数已经可以完成字节序的值拷贝了,不需要自己显式实现。这是因为默认拷贝构造函数会逐个复制对象的所有非静态成员变量,包括简单类型(如 int、double 等)和数组等。

然而,当一个类拥有指针或引用等需要特别注意的成员变量时,编译器生成的默认拷贝构造函数不能保证正确的深拷贝,会导致浅拷贝问题和内存泄漏等问题。此时,需要手动定义一个拷贝构造函数来进行深拷贝操作,从而避免这些问题的出现。

因此,需要根据具体情况来决定是否需要自己显式实现拷贝构造函数。如果类中只有简单类型的成员变量,就可以使用编译器生成的默认拷贝构造函数;如果类中有指针或引用等需要特别注意的成员变量,就需要手动实现一个深拷贝的拷贝构造函数。

这里会发现下面的程序会崩溃掉?这里就需要我们用深拷贝去解决。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
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);
 s1.Push(3);
 s1.Push(4);
 Stack s2(s1);
 return 0;
}

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

对象拷贝

在C++中,对象拷贝指的是将一个对象的值复制到另一个对象中。常见的对象拷贝方法包括拷贝构造函数赋值运算符

拷贝构造函数是用来创建一个对象,该对象与另一个对象具有相同的值。它通常用于实现深拷贝,并且可以从其他对象中创建一个新对象。拷贝构造函数的语法如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class MyClass {
public:
  MyClass(const MyClass& other); // 拷贝构造函数
};

其中 other 是要拷贝的对象的引用。

赋值运算符是用于将一个对象的值复制到另一个对象中的运算符。通常使用 = 符号进行赋值操作。赋值运算符的语法如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class MyClass {
public:
  MyClass& operator=(const MyClass& other); // 赋值运算符
};

其中 other 是要拷贝的对象的引用。注意赋值运算符返回值为当前对象的引用,以支持链式赋值操作。

需要注意的是,对象拷贝可能涉及浅拷贝和深拷贝的概念,因此需要根据情况选择适当的拷贝方法。如果类中包含指针或资源管理的成员变量,则需要手动实现深拷贝,以确保正确的对象复制和资源释放。否则,在执行浅拷贝时,两个对象将共享同一块内存,可能会导致悬挂指针、内存泄漏等问题。

在使用对象拷贝时,还需要注意对象的生命周期和内存管理,避免出现悬挂指针、内存泄漏等问题。

浅拷贝:

浅拷贝是指简单地将一个对象的值复制给另一个对象,包括对象中的所有成员变量。这意味着拷贝后的对象和原始对象共享同一块内存,当其中一个对象修改了内存中的值时,另一个对象也会受到影响。这种情况下,如果两个对象的析构函数试图同时释放同一块内存,会导致内存错误。

深拷贝:

深拷贝是指创建一个对象的独立副本,其中包括对象中的所有成员变量。这意味着拷贝后的对象拥有自己的内存空间,对其中一个对象的修改不会影响另一个对象。这种情况下,每个对象的析构函数可以安全地释放自己拥有的内存。

为了实现深拷贝,通常需要手动分配内存并将原始对象中的数据复制到新对象中,例如使用 new 运算符来动态分配内存,并通过拷贝构造函数或赋值运算符将数据复制到新对象中。而浅拷贝则可以使用默认的拷贝构造函数和赋值运算符,由编译器自动生成。

需要特别注意的是,如果类中包含指针或资源管理的成员变量(如动态分配的内存),则需要手动实现深拷贝以确保正确的对象复制和资源释放。否则,在执行浅拷贝时,两个对象将共享同一块内存,可能会导致悬挂指针、内存泄漏等问题。

因此,当类中存在指针或资源管理的成员变量时,通常需要自定义拷贝构造函数和赋值运算符,以实现深拷贝,避免出现潜在的问题。

示例理解:

假如现在你买了(构造)一个房子,开发商给你配了一个钥匙(指针成员变量)

那么,现在你的朋友也行买个和你一样的房子,也声明了另一个房子,然后(拷贝构造)

这时我们发现系统崩溃了,为什么呢?而且我们可以发现运行的出来的地址是一样的,这证明两个人的钥匙配对的是同一套“房子”,所以这是错误的!因为C++不知道你复制一把钥匙的目的是什么,所以就只是单纯的复制了一把钥匙,这就是浅拷贝!

我们要效果是,你的朋友要的是和你有一样的房子,而不是同一个,所以我们自定义一个拷贝构造函数,这时的运行结果显示,两套房子的地址不一样了~这就是深拷贝!

析构函数析构完后意味着“第一队拆迁办”已经把第一套房子拆了,而此时两个钥匙指向的同一套房子,当“第二队拆迁办”来了之后,发现,好家伙,房子已经被拆了,所以程序就报错了!!

技术总结:

C++默认生成的拷贝构造函数,他的行为就是浅拷贝,他只会复制一个一摸一样的指针,并不会操作指针指向的东西。要想实现我们的逻辑需求,就要自定义拷贝构造函数,实现深拷贝。

5. 拷贝构造函数典型调用场景:
  • 使用已存在对象创建新对象
  • 函数参数类型为类类型对象
  • 函数返回值类型为类类型对象
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class Date
{
public:
	Date(int year, int minute, int day)
	{
		cout << "Date(int,int,int):" << this << endl;
	}
	Date(const Date& d)
	{
		cout << "Date(const Date& d):" << this << endl;
	}
	~Date()
	{
		cout << "~Date():" << this << endl;
	}
private:
	int _year;
	int _month;
	int _day;
};
Date Test(Date d)
{
	Date temp(d);
	return temp;
}
int main()
{
	Date d1(2022, 1, 13);
	Test(d1);
	return 0;
}

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

希望对你有帮助!加油!

若您认为本文内容有益,请不吝赐予赞同并订阅,以便持续接收有价值的信息。衷心感谢您的关注和支持!

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2024-05-23,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
C++奇迹之旅:深入思考拷贝构造函数
拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动调用。
学习起来吧
2024/04/20
1220
C++奇迹之旅:深入思考拷贝构造函数
【C++】拷贝构造函数和赋值运算符重载详解
拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存 在的类类型对象创建新对象时由编译器自动调用。 拷贝构造函数是一个特殊的构造函数,用于创建一个新的对象,其内容与另一个已存在的对象相同。在C++中,拷贝构造函数通常用于将一个对象的值复制到另一个对象中(一个对象存在,一个对象不存在),以便在程序中进行对象的赋值和传递操作时,能够确保对象的内容被正确复制。
P_M_P
2024/02/05
2580
【C++】拷贝构造函数和赋值运算符重载详解
【c++】类和对象(四)深入了解拷贝构造函数
拷贝构造函数是构造函数的一个重载形式,拷贝构造函数的参数只有一个且必须是类类型对象的引用,使用传值方式编译器直接报错,因为会引发无穷递归调用,这个我们后面进行讲解
用户11029103
2024/04/02
1160
【c++】类和对象(四)深入了解拷贝构造函数
【C++】拷贝构造函数与赋值的区别
将一个类拷贝到另一个类中,自然是需要对应的参数的,参数就是类,这里可以传递指针,引用更好,只是传值形参的话,会造成无限递归。
啊QQQQQ
2024/11/19
1150
【C++】拷贝构造函数与赋值的区别
【C++类和对象】拷贝构造与赋值运算符重载
拷贝构造函数:拷贝构造是指在创建一个新对象时,使用已存在的对象作为其初始值的构造函数。只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动调用。
大耳朵土土垚
2024/04/20
1970
【C++类和对象】拷贝构造与赋值运算符重载
【C++ 类和对象 进阶篇】—— 逻辑森林的灵动精灵,舞动类与对象的奇幻圆舞曲
类的默认成员函数是编译器在没有显式定义相应函数时自动生成的函数。这些函数通常是为了处理类对象的生命周期管理,包括对象的创建、复制、赋值和销毁等操作。确保即使开发者没有显式提供某些操作,编译器也能提供默认实现,以保证程序的基本功能。 通常包括以下几个函数:
换一颗红豆
2025/01/24
430
【C++ 类和对象 进阶篇】—— 逻辑森林的灵动精灵,舞动类与对象的奇幻圆舞曲
【C++修炼之路】C++类类和对象进一步探索,六个幕后英雄
在上一篇《C++类与对象入门:从封装到this指针的初探》中,我们学习了如何定义类、创建对象,并通过封装保护数据。然而,类的真正力量远不止于此——
f狐o狸x
2025/03/04
610
【C++修炼之路】C++类类和对象进一步探索,六个幕后英雄
[C++]类与对象中篇
类与对象中篇:: 1.类的6个默认成员函数 如果一个类中什么成员都没有,简称为空类。空类真的什么都没有吗?并不是,任何类在什么都不写时,编译器会自动生成6个默认成员函数。 默认成员函数: 用户没有显示
IT编程爱好者
2023/04/12
5260
[C++]类与对象中篇
C++第四弹 -- 类与对象中篇上(构造函数 析构函数 拷贝构造函数)
让我们一起揭开 C++ 对象生命周期管理的神秘面纱,掌握构造函数、析构函数和拷贝构造函数的精髓!
用户11317877
2024/10/16
1160
C++第四弹 -- 类与对象中篇上(构造函数 析构函数 拷贝构造函数)
【C++篇】C++类与对象深度解析(三):类的默认成员函数详解
运算符重载允许我们为类对象自定义运算符的行为,这样当我们对类对象使用这些运算符时,它们会按照我们定义的方式执行。如果没有定义对应的运算符重载,编译器将会报错,因为它不知道如何处理这些运算符。
半截诗
2024/10/09
1340
【C++篇】C++类与对象深度解析(三):类的默认成员函数讲解
4.1 运算符重载的基本概念 运算符重载允许我们为类对象自定义运算符的行为,这样当我们对类对象使用这些运算符时,它们会按照我们定义的方式执行。如果没有定义对应的运算符重载,编译器将会报错,因为它不知道如何处理这些运算符。
熬夜学编程的小王
2024/11/20
660
【C++篇】C++类与对象深度解析(三):类的默认成员函数讲解
【C++】踏上C++学习之旅(八):深入“类和对象“世界,掌握编程的黄金法则(三)(内含运算符重载和拷贝构造函数)
在之前的文章中,相信大家已经对"类"这个面向对象的语法以及一些基本的用法已经掌握了,那么在本文中将会带着大家继续解读,"类和对象"世界别致的风景——“拷贝构造函数"和"赋值运算符重载”。当然还有精彩的运算符重载语法讲解哦~
埋头编程
2024/11/21
840
【C++】踏上C++学习之旅(八):深入“类和对象“世界,掌握编程的黄金法则(三)(内含运算符重载和拷贝构造函数)
【C++修行之道】类和对象(三)拷贝构造函数
四、编译器生成的默认拷贝构造函数已经可以完成字节序的值拷贝了,还需要自己显式实现吗?
走在努力路上的自己
2024/05/31
960
【C++修行之道】类和对象(三)拷贝构造函数
C++的六大“天选之子“拷贝构造与与运算符重载
假设哦我们需要创建两个一模一样的对象A和B. 那我们可以先创建一个对象A,再通过将A作为参数,传给B进行初始化, 即一个自定义类型实例化出的对象(B)用另一个该类型实例化出的对象(A)进行初始化.
初阶牛
2023/10/14
1780
C++的六大“天选之子“拷贝构造与与运算符重载
C++从入门到精通——类的6个默认成员函数之拷贝构造函数
空类中真的什么都没有吗?并不是,任何类在什么都不写时,编译器会自动生成以下6个默认成员函数。
鲜于言悠
2024/04/16
3370
C++从入门到精通——类的6个默认成员函数之拷贝构造函数
C++新旅程:类的拷贝构造函数 、赋值运算符重载 和const成员函数
https://cloud.tencent.com/developer/article/2466159?shareByChannel=link
池央
2024/11/23
980
C++新旅程:类的拷贝构造函数 、赋值运算符重载 和const成员函数
【C++】C++入门—初识构造函数 , 析构函数,拷贝构造函数,赋值运算符重载
如果一个类中什么成员都没有,简称为空类。 空类中真的什么都没有吗? 并不是 任何类在什么都不写时,编译器会自动生成以下6个默认成员函数。 默认成员函数:用户没有显式实现,编译器会生成的成员函数称为默认成员函数 我们实现了,编译器就不会生成了
叫我龙翔
2024/02/19
1830
【C++】C++入门—初识构造函数 , 析构函数,拷贝构造函数,赋值运算符重载
C++进阶之路:何为运算符重载、赋值运算符重载与前后置++重载(类与对象_中篇)
C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似。
Srlua
2024/05/26
1060
C++进阶之路:何为运算符重载、赋值运算符重载与前后置++重载(类与对象_中篇)
类与对象(中(2))
大家好啊,上一期内容我们介绍了类与对象中六大默认成员函数中的两种--->构造函数与析构函数,相信大家多少都形成了自己的独到见解。那么今天,我将继续就拷贝构造函数与运算符重载函数来展开讲解,话不多说,我们进入正题~~
用户11289931
2024/09/24
830
类与对象(中(2))
【C++篇】C++类与对象深度解析(二):类的默认成员函数详解
在C++中,默认成员函数是指用户没有显式实现,而由编译器自动生成的成员函数。一个类在没有显式定义特定成员函数的情况下,编译器会自动生成以下6个默认成员函数。理解这些默认成员函数的行为和作用是掌握C++类机制的基础。
半截诗
2024/10/09
2050
【C++篇】C++类与对象深度解析(二):类的默认成员函数详解
推荐阅读
C++奇迹之旅:深入思考拷贝构造函数
1220
【C++】拷贝构造函数和赋值运算符重载详解
2580
【c++】类和对象(四)深入了解拷贝构造函数
1160
【C++】拷贝构造函数与赋值的区别
1150
【C++类和对象】拷贝构造与赋值运算符重载
1970
【C++ 类和对象 进阶篇】—— 逻辑森林的灵动精灵,舞动类与对象的奇幻圆舞曲
430
【C++修炼之路】C++类类和对象进一步探索,六个幕后英雄
610
[C++]类与对象中篇
5260
C++第四弹 -- 类与对象中篇上(构造函数 析构函数 拷贝构造函数)
1160
【C++篇】C++类与对象深度解析(三):类的默认成员函数详解
1340
【C++篇】C++类与对象深度解析(三):类的默认成员函数讲解
660
【C++】踏上C++学习之旅(八):深入“类和对象“世界,掌握编程的黄金法则(三)(内含运算符重载和拷贝构造函数)
840
【C++修行之道】类和对象(三)拷贝构造函数
960
C++的六大“天选之子“拷贝构造与与运算符重载
1780
C++从入门到精通——类的6个默认成员函数之拷贝构造函数
3370
C++新旅程:类的拷贝构造函数 、赋值运算符重载 和const成员函数
980
【C++】C++入门—初识构造函数 , 析构函数,拷贝构造函数,赋值运算符重载
1830
C++进阶之路:何为运算符重载、赋值运算符重载与前后置++重载(类与对象_中篇)
1060
类与对象(中(2))
830
【C++篇】C++类与对象深度解析(二):类的默认成员函数详解
2050
相关推荐
C++奇迹之旅:深入思考拷贝构造函数
更多 >
领券
社区富文本编辑器全新改版!诚邀体验~
全新交互,全新视觉,新增快捷键、悬浮工具栏、高亮块等功能并同时优化现有功能,全面提升创作效率和体验
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文