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

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


拷贝构造函数

概念 :

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

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

特征:

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

1. 拷贝构造函数是构造函数的一个重载形式

#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 运算符,在堆上分配新的内存,并将原对象的数据复制到新内存中。

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

obj1 data: 42
obj2 data: 42

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


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;
}

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;
}

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

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

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

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

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

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

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++中,对象拷贝指的是将一个对象的值复制到另一个对象中。常见的对象拷贝方法包括拷贝构造函数赋值运算符

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

class MyClass {
public:
  MyClass(const MyClass& other); // 拷贝构造函数
};

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

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

class MyClass {
public:
  MyClass& operator=(const MyClass& other); // 赋值运算符
};

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

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

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

浅拷贝:

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

深拷贝:

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

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

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

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

示例理解:

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

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

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

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

技术总结:

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

5. 拷贝构造函数典型调用场景:

  • 使用已存在对象创建新对象
  • 函数参数类型为类类型对象
  • 函数返回值类型为类类型对象
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;
}

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

希望对你有帮助!加油!

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

目录
相关文章
|
3月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
1月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
58 12
|
2月前
|
设计模式 安全 C++
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
59 16
|
2月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
2月前
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
2月前
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
152 6
|
2月前
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
3月前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
4月前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
127 19
|
3月前
|
安全 编译器 C语言
【C++篇】深度解析类与对象(中)
在上一篇博客中,我们学习了C++类与对象的基础内容。这一次,我们将深入探讨C++类的关键特性,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载、以及取地址运算符的重载。这些内容是理解面向对象编程的关键,也帮助我们更好地掌握C++内存管理的细节和编码的高级技巧。

热门文章

最新文章