[C++]构造与毁灭:深入探讨C++中四种构造函数与析构函数

  •  个人主页:北·海
  •  🎐CSDN新晋作者
  •  🎉欢迎 👍点赞✍评论⭐收藏
  • ✨收录专栏:C/C++
  • 🤝希望作者的文章能对你有所帮助,有不足的地方请在评论区留言指正,大家一起学习交流!🤗

目录

构造函数有什么作用?

构造函数有什么特点

构造函数的种类

一.默认构造函数

1.什么是默认构造函数

2.默认构造函数的应用

使用情况一:类内初始值

使用情况二:创建对象数组

使用情况三:在派生类中

自定义的默认构造函数

 二.自定义的重载构造函数

1.构造函数的作用

三.拷贝构造函数

1.浅拷贝

2.深拷贝

3.什么时候用到深拷贝/浅拷贝

4.什么时候会调用拷贝构造函数

 四.赋值构造函数

1.赋值构造函数可以怎么样定义

2.赋值构造函数在什么时候会调用?

2.赋值构造函数与拷贝构造函数的区别

五.析构函数

1.析构函数的基本概念

2.容易将调用析构函数与delete释放内存混淆


 概要:

"构造与毁灭,是C++中对象生命周期的两个重要阶段。构造函数用于初始化对象的状态和数据成员,为对象提供合适的初始值;而析构函数则在对象销毁时执行清理工作,释放资源,确保对象的安全结束。通过合理设计构造函数和析构函数,我们能够在对象的创建和销毁过程中维护良好的程序行为和资源管理。


构造函数有什么作用?

        在创建一个新的对象时,自动调用的函数,用来进行“初始化”工作:对这个对象内部的数据成员进行初始化

构造函数有什么特点

  1. 自动调用(在创建新对象时,自动调用)
  2. 构造函数的函数名,和类名相同
  3. 构造函数没有返回类型
  4. 可以有多个构造函数(即函数重载形式)

构造函数的种类

     1. 默认构造函数

     2. 自定义的构造函数

     3.拷贝构造函数

     4.赋值构造函数


一.默认构造函数

1.什么是默认构造函数

    1.没有参数的构造函数成为默认构造函数

    2.没有手动定义默认构造函数时,编译器自动为这个类定义一个构造函数。

    3. 如果数据成员使用了“类内初始值”,就使用这个值来初始化数据成员。否则,就使用默认初始化(实际上,不做任何初始化)只有C++11可以使用类内初始值】

    

2.默认构造函数的应用

  使用情况一:类内初始值

   以上是当全部成员变量有"类内初始值"时,可以使用默认构造函数

   使用情况二:创建对象数组

      由此报错可以看出,对象数组的创建,必须使用默认的构造函数,在上面87行写了自定义的构造函数之后,程序就不会默认生成默认构造函数了,要想在有自定义构造函数的情况下,创建对象数组,就必须再重载一个不含参数的默认构造函数

使用情况三:在派生类中

当派生类没有显式定义构造函数时,它将继承基类的默认构造函数。这使得在创建派生类对象时,基类的成员变量可以正确地初始化。

#include <iostream>
using namespace std;

// 基类
class Base {
public:
    int baseValue;

    // 默认构造函数
    Base() {
        baseValue = 0;
        cout << "Base 默认构造函数被调用" << endl;
    }
};

// 派生类
class Derived : public Base {
public:
    int derivedValue;

    void printValues() {
        cout << "Base 值: " << baseValue << endl;
        cout << "Derived 值: " << derivedValue << endl;
    }
};

int main() {
    Derived derivedObj;
    derivedObj.derivedValue = 10;
    derivedObj.printValues();
    return 0;
}


输出:
Base 默认构造函数被调用
Base 值: 0
Derived 值: 10

可以看到,派生类的对象 derivedObj 在创建时成功继承了基类 Base 的默认构造函数,并正确地初始化了基类的成员变量 baseValue。这证实了在派生类中没有显式定义构造函数时,它将继承基类的默认构造函数,并能够正确初始化基类的成员变量。

注意:

只要手动定义了任何一个构造函数,编译器就不会生成“默认构造函数”

一般情况下,都应该定义自己的构造函数,不要使用“默认构造函数”

【仅当数据成员全部使用了“类内初始值”,才宜使用“合成的默认构造函数”】

自定义的默认构造函数

如果既有类内初始值,在默认构造函数里面又有初始化,则以默认构造函数里的为准

说明:如果某数据成员使用类内初始值,同时又在构造函数中进行了初始化,那么以构造函数中的初始化为准。相当于构造函数中的初始化,会覆盖对应的类内初始值。


 二.自定义的重载构造函数

1.构造函数的作用

  1. 初始化对象:自定义构造函数允许你在创建对象时对其进行初始化。您可以通过构造函数的参数传递初始值,或在构造函数中执行特定的初始化操作,确保对象在创建时处于正确的状态。

  2. 参数化构造:自定义构造函数可以接受参数,从而使对象的创建更加灵活和可定制化。通过不同的构造函数形式,可以为不同的使用场景提供不同的对象初始化方式。

  3. 代码可读性和维护性:通过显式定义构造函数,可以提高代码的可读性和维护性。构造函数明确地指示了对象的创建方式和初始化过程,使代码更加清晰易懂,并且便于后续修改维护。

// 定义一个“人类”
class Human {
public:  
	Human();
	Human(int age, int salary);//自定义构造函数

	string getName();
	int getAge();
	int getSalary();

private:
	string name = "Unknown";
	int age = 28;
	int salary;
};

Human::Human() {
	name = "无名氏";
	age = 18;
	salary = 30000;
}

Human::Human(int age, int salary) {
	cout << "调用自定义的构造函数" << endl; 
	this->age = age;      //this是一个特殊的指针,指向这个对象本身
	this->salary = salary;
	name = "无名";
}



string Human::getName() {
	return name;
}

int Human::getAge() {
	return age;
}

int Human::getSalary() {
	return salary;
}


int main(void) {
	Human  h1(25, 35000);  // 使用自定义的默认构造函数

	cout << "姓名:" << h1.getName() << endl;
	cout << "年龄: " << h1.getAge() << endl;    
	cout << "薪资:" << h1.getSalary() << endl; 

	system("pause");
	return 0;
}

由此可以看出,自定义构造函数,只是比默认构造函数多了形参,由于创建的对象都形形色色,所以大多数情况下都会用自定义构造函数,在定义对象时,将参数传递进去给对象赋值


三.拷贝构造函数

1.浅拷贝

浅拷贝是指在对象拷贝过程中,仅简单地复制对象的成员变量值,而不复制成员变量所指向的动态分配的内存。这意味着原对象和拷贝对象将共享同一块内存区域,对其中一个对象的修改会影响到另一个对象。

在进行浅拷贝时,通常会使用默认的拷贝构造函数或赋值运算符重载来完成。这些默认的复制操作只会简单地逐个拷贝对象的成员变量的值。

class Human {
private:
	int Age=30;
	string Name ="LiHua";
public:
	Human(int age,string name);
	int getAge();
	string getName();

	void print();
};


int Human::getAge() {
	return this->Age;
}

string Human::getName() {
	return this->Name;
}

void Human::print() {
	cout << "姓名 : " << this->getName() << endl;
	cout << "年龄 : " << this->getAge() << endl;
}
Human::Human(int age,string name) {
	this->Age = age;
	this->Name = name;
}
int main() {

	Human h1(19,"Lihua");
	Human h2 = h1;//给对象赋值方法一
	Human h3(h2);//给对象赋值方法二

	h1.print();
	h2.print();
	h3.print();
}

由此可以得出:给对象的赋值方法有两种,这两种方法都会调用拷贝构造函数,由于三个变量用的一块内存,所以说,改其中一个对象的值,其他三个的值都会被改变

上面两张图片中输出的地址是一个地址,改变一个对象的值,另一个对象的值也会改变,这就验证了,浅拷贝的结果是两个对象共用了一个地址

说明 : 默认的拷贝构造函数的缺点: 使用“浅拷贝”

2.深拷贝

根据上面浅拷贝中的例子,其中有个地址,如果h1想要改addr,但是h2却不想改他的addr,这种情况就需要用到深拷贝了,深拷贝就必须得写自定义的拷贝构造函数,例子如下:

class Human {
private:
	int Age=30;
	string Name ="LiHua";
    char* addr;
public:
	Human(int age,string name);
	Human(const Human& the);
    ~Human();//析构函数

	int getAge();
	string getName();

	void setAge(int age);
	void setName(string name);
	void setAddr(const char* addr);
	
	void print();

};
int Human::getAge() {
	return this->Age;
}

string Human::getName() {
	return this->Name;
}

Human::~Human(){

delete addr;
}

void Human::print() {
	cout << "姓名 : " << this->getName() << endl;
	cout << "年龄 : " << this->getAge() << endl;
	cout << "地址 : " << this->addr << endl;
	printf("该对象的addr地址 : %p\n", this->addr);
}
Human::Human(int age,string name) {
	this->Age = age;
	this->Name = name;
	addr = new char[64];
	strcpy(addr, "China");
}
void Human::setName(string name) {
	this->Name = name;
}
void Human::setAge(int age) {
	this->Age = age;
}
void Human::setAddr(const char* addr) {
	if (!addr) {
		return;
	}
	strcpy(this->addr, addr);
}
Human::Human(const Human& the) {
	this->Age = the.Age;
	this->Name = the.Name;
	//给拷贝对象的addr重新分配内存,存储该地址
	this->addr = new char[64];
	strcpy(this->addr, the.addr);
}
int main() {

	Human h1(19,"Lihua");
	Human h2(h1);//给对象赋值方法一
	//Human h3=h1;//给对象赋值方法二

	h1.print();
	h2.print();
	//addr是char*类型
	h2.setAddr("美国");
	cout << "------------------------" << endl;
	h1.print();
	h2.print();
}

可以看到,此时通过深拷贝,两个对象的addr地址已经改变了

3.什么时候用到深拷贝/浅拷贝

  1. 对象拥有动态分配的资源:如果对象包含了动态分配的内存(如使用 newmalloc 创建的内存),在进行拷贝时需要进行深拷贝。这是因为默认的浅拷贝(shallow copy)只会复制指针,而不会为新对象分配独立的内存空间,这可能会导致多个对象指向同一内存,造成资源释放问题或不可预测的行为。

  2. 修改的独立性要求:如果你需要在拷贝对象后对其进行修改,而不希望修改影响到原始对象,那么深拷贝是必需的。深拷贝会创建一个完全独立的副本,修改副本不会影响原始对象。

  3. 对象包含指向其他对象的引用:当一个对象包含指向其他对象的引用或指针时,进行拷贝时可能需要进行深拷贝。这样可以确保每个对象都有自己的引用,而不是共享同一个引用。

需要注意的是,并非所有情况都需要进行深拷贝。有时候浅拷贝已经足够满足需求,尤其是当拷贝的对象是只读的或者没有包含指向动态分配资源的指针时。

在 C++ 中,可以通过自定义拷贝构造函数和赋值运算符重载来实现深拷贝

4.什么时候会调用拷贝构造函数

1.调用函数时,实参是对象,形参不是引用类型

2.函数的返回类型是类.而且不是引用类型

3.对象数组的初始化列表中,使用对象 


 四.赋值构造函数

1.赋值构造函数可以怎么样定义

 使用重载运算符 " = "进行实现

class Human {
private:
	int age;
	string name;
	char* addr;

public:
	//通过重载 " = "实现
	Human& operator=(const Human& other);
	
	Human(int age, string name,const char*addr);
	Human(){}
	~Human() {
		cout << "调用析构函数" << endl;
		delete addr;
	}

	void print() {
		cout << "age :" << age << endl;
		cout << "name : " << name << endl;
		cout << "addr : " << addr << endl;
	}
};

Human& Human::operator=(const Human& other) {
	if (this == &other) {
		return *this;
	}

	addr = new char[64];
	strcpy(addr, other.addr);

	age = other.age;
	name = other.name;

	return *this;
}
Human::Human(int age, string name, const char* addr) :age(age), name(name) {
	this->addr = new char[64];
	strcpy(this->addr, addr);
}

int main() {
	Human h1(23, "LiHua", "Chinese"),h2;
	h2 = h1;
	h1.print();
	cout << "************" << endl;
	h2.print();

}

2.赋值构造函数在什么时候会调用?

当用一个对象给另一个对象进行赋值时候会被调用,定义加赋值的话调用拷贝构造函数,例如:

int main(){

Test p1(20);
Test p2 = p1;
此时会调用拷贝构造函数
}
class Test{

.....
public:

Test(int n){
this.age  =20;
}
Test test(Test & man){
//内联函数
return man;//返回对象
}

};

int main(){

Test p1(20),p2;
p2 = p1;//此时会调用赋值构造函数

Test p3 = h1;//此时创建对象p3同时初始化,会调用的是拷贝构造函数 

p2 = test(p1);//此时会调用赋值构造函数

Test p4  =test(p1);//此时会调用拷贝构造函数

}

2.赋值构造函数与拷贝构造函数的区别

  1. 触发时机:拷贝构造函数在创建一个新对象并初始化时被调用,而赋值构造函数在已存在的对象进行赋值操作时被调用。

  2. 参数类型:拷贝构造函数使用被拷贝对象的引用作为参数,通常是常量引用,用于初始化新对象;而赋值构造函数使用所需赋值的对象的引用作为参数,用于将已存在的对象赋值给另一个已存在的对象。

  3. 功能:拷贝构造函数的主要目的是创建一个新对象,并将其初始化为与被拷贝对象相同的值。它通常用于深拷贝,确保新对象与原对象是独立的。赋值构造函数的主要目的是将一个已经存在的对象的值赋给另一个已经存在的对象。

  4. 默认实现:如果没有显式定义拷贝构造函数和赋值构造函数,C++ 编译器会为类生成默认的拷贝构造函数和默认的赋值构造函数。默认的拷贝构造函数会执行浅拷贝,简单地将成员变量的值复制给新对象。默认的赋值构造函数也执行浅拷贝,将每个成员变量的值从一个对象复制到另一个对象。

需要明确的是,拷贝构造函数和赋值构造函数在语法上是不同的,它们具有不同的函数形参和使用方式。在设计类时,根据对象的需求,需要选择正确的构造函数来满足对象的初始化和赋值操作。


五.析构函数

1.析构函数的基本概念

作用:

对象销毁前,做清理工作。

具体的清理工作,一般和构造函数对应

比如:如果在构造函数中,使用new分配了内存,就需在析构函数中用delete释放。

如果构造函数中没有申请资源(主要是内存资源),

那么很少使用析构函数。

函数名:

~类型

没有返回值,没有参数,最多只能有一个析构函数

访问权限:

一般都使用public

使用方法:

不能主动调用。

对象销毁时,自动调用。

如果不定义,编译器会自动生成一个析构函数(什么也不做)

2.容易将调用析构函数与delete释放内存混淆

在我接触析构函数的时候,一直懂得就是在对象被销毁的时候才会调用析构函数,但是每次在类的成员函数中遇到new出来的空间时,就会以为将delete写在该成员函数中,然后delete执行了就会去调用析构函数,这个从逻辑上都是说不过去的

现在懂得概念很明确,只有在对象销毁的时候调用,在类的成员函数或者构造函数中遇到new出来的空间时候,就应该给析构函数里面写delete释放该空间,等到对象被销毁的时候,就会去调用析构函数,就会执行delete语句将内存释放

现在来看这么一个简单的例子,输出的顺序,就明白,只有对象被销毁的时候才会调用析构函数,遇见了new,就应该在析构函数里面写delete

class Human {
private:
	int age;
	string name;
	char* addr;

public:
	Human() {
		cout << "调用构造函数" << endl;
		addr = new char[64];
	}

	~Human() {
		cout << "调用析构函数" << endl;

		delete[]this->addr;
	}

	void print() {
		cout << "调用print函数" << endl;
		strcpy(addr, "China");
		cout << addr << endl;
	}
};

int main() {
	Human p1;
	p1.print();
}


总结:

"在C++中,构造函数和析构函数是类的特殊成员函数,它们扮演着至关重要的角色。通过构造函数,我们可以初始化对象的状态,确保对象在被创建时处于合适的状态。而析构函数则负责在对象生命周期结束时进行善后工作,释放动态分配的资源,保证对象的安全销毁。深入理解和合理设计构造函数和析构函数,可以帮助我们编写更可靠、高效的C++程序,有效地管理资源,避免内存泄漏和访问冲突。构造与毁灭是C++编程中的关键概念,它们共同构成了面向对象编程中的基石,为我们提供了强大而灵活的工具,让我们能够更好地利用和管理对象的生命周期。" 


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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

05架构管理之持续集成-DevOps的理解与实现

专栏说明&#xff1a;针对于企业的架构管理岗位&#xff0c;分享架构管理岗位的职责&#xff0c;工作内容&#xff0c;指导架构师如何完成架构管理工作&#xff0c;完成架构师到架构管理者的转变。计划以10篇博客阐述清楚架构管理工作&#xff0c;专栏名称&#xff1a;架构管理…

分享一个vue-slot插槽使用场景

需求再现 <el-table-column align"center" label"状态" prop"mitStatus" show-overflow-tooltip />在这里&#xff0c;我想对于状态进行一个三目判断&#xff0c;如果为0那就是进行中&#xff0c;否则就是已完成&#xff0c;期初我是这样写…

项目-IM

zk 启动类实现CommandLineRunner接口&#xff0c;重写run()方法 单聊 群聊 离线消息

Android OTA 相关工具(六) 使用 lpmake 打包生成 super.img

我在 《Android 动态分区详解(二) 核心模块和相关工具介绍》 介绍过 lpmake 工具&#xff0c;这款工具用于将多个分区镜像打包生成一个 Android 专用的动态分区镜像&#xff0c;一般称为 super.img。Android 编译时&#xff0c;系统会自动调用 lpmake 并传入相关参数来生成 sup…

uniapp实现:点击拨打电话,弹出电话号码列表,可以选择其中一个进行拨打

一、实现效果&#xff1a; 二、代码实现&#xff1a; 在uni-app中&#xff0c;使用uni.showActionSheet方法实现点击拨打电话的功能&#xff0c;并弹出相关的电话列表供用户选择。 当用户选择了其中一个电话后&#xff0c;会触发success回调函数&#xff0c;并通过res.tapInde…

ELK日志收集系统

目录 一、概述 二、组件 一、logstash 一、工作过程 二、INPUT 三、FILETER 四、OUTPUTS 二、elasticsearch 三、kibana 三、架构类型 一、ELK 二、ELKK 三、ELFK 四、ELFKK 五、EFK 四、配置ELK日志收集系统集群实验的步骤文档 五、配置ELK日志收集系统集群 …

0201hdfs集群部署-hadoop-大数据学习

文章目录 1 前言2 集群规划3 hadoop安装包上传与安装3.1 上传解压 4 hadoop配置5 从节点同步和环境变量配置6 创建用户7 集群启动8 问题集8.1 Invalid URI for NameNode address (check fs.defaultFS): file:/// has no authority. 结语 1 前言 下面我们配置下单namenode节点h…

地下管线三维自动建模软件MagicPipe3D V3.0发布

2023年9月1日经纬管网建模系统MagicPipe3D V3.0正式发布&#xff0c;该版本经过众多用户应用和反馈&#xff0c;在三维地下管网建模效果、效率、适配性方面均有显著提升&#xff01;MagicPipe3D本地离线参数化构建地下管网模型&#xff08;包括管道、接头、附属设施等&#xff…

GPT能否辅助数学学习

GPT4.0的数学能力怎么样&#xff1f;我们使用镜像站进行实验&#xff0c;通过不同水平的数学看看GPT4.0的数学能力得到进步没有。镜像站的地址我们放在了最后&#xff0c;各位读者也可以自行去测试。 笔者在ChatGPT镜像站进行测试&#xff0c;我们的实验是让GPT4.0自行出数学题…

[javaWeb]Socket网络编程

网络编程&#xff1a;写一个应用程序,让这个程序可以使用网络通信。这里就需要调用传输层提供的 api。 Socket套接字 传输层提供协议&#xff0c;主要是两个: UDP和TCP 提供了两套不同的 api&#xff0c;这api也叫做socket api。 UDP和 TCP 特点对比&#xff1a; UDP: 无连…

【Redis从头学-13】Redis哨兵模式解析以及搭建指南

&#x1f9d1;‍&#x1f4bb;作者名称&#xff1a;DaenCode &#x1f3a4;作者简介&#xff1a;啥技术都喜欢捣鼓捣鼓&#xff0c;喜欢分享技术、经验、生活。 &#x1f60e;人生感悟&#xff1a;尝尽人生百味&#xff0c;方知世间冷暖。 &#x1f4d6;所属专栏&#xff1a;Re…

安防监控/视频汇聚平台EasyCVR调用rtsp地址返回的IP不正确是什么原因?

安防监控/云存储/磁盘阵列存储/视频汇聚平台EasyCVR可拓展性强、视频能力灵活、部署轻快&#xff0c;可支持的主流标准协议有GB28181、RTSP/Onvif、RTMP等&#xff0c;以及厂家私有协议与SDK接入&#xff0c;包括海康Ehome、海大宇等设备的SDK等&#xff0c;能对外分发RTSP、RT…

QML Book 学习基础4(状态和转换)

目录 states&#xff08;状态&#xff09; Transition&#xff08;过渡&#xff09; states&#xff08;状态&#xff09; 用户界面的某些部分可以用状态来描述。状态定义一组属性更改&#xff0c;并且可以由特定条件触发。 QML 中定义状态&#xff0c;该元素需要绑定到任何项…

JavaScript中的事件委托(event delegation)

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ JavaScript事件委托⭐ 事件冒泡&#xff08;Event Bubbling&#xff09;⭐ 事件委托的优点⭐ 如何使用事件委托⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 记得点击上方或者右侧链接订阅本专栏哦 几何带你启…

RuntimeError: scatter_cpu_(): Expected self.dtype to be equal to src.dtype

1. 问题描述 如下图&#xff0c;输入scatter_时报错&#xff01; 2. 报错原因 查阅资料发现是因为要填充的value与要被填充的tensor类型不同&#xff01;如下图 3. 解决办法 将其转换成一样的类型即可&#xff0c;如下图&#xff0c;测试没有报错&#xff1a;

深度学习怎么学?

推荐这本小白看的《深度学习&#xff1a;从基础到实践&#xff08;上下册&#xff09;》。 深度学习&#xff1a;从基础到实践&#xff08;上下册&#xff09; 深入浅出的讲述了深度学习的基本概念与理论知识&#xff0c;不涉及复杂的数学内容&#xff0c;零基础小白也能轻松掌…

为什么删除Windows 11上的Bloatware可以帮助加快你的电脑速度

如果你感觉你的电脑迟钝&#xff0c;彻底清除软件会有所帮助&#xff0c;而且这个过程对Windows用户来说越来越容易。 微软正在使删除以前难以删除的其他预装Windows应用程序成为可能。专家表示&#xff0c;这项新功能可能会改变用户的游戏规则。 科技公司Infatica的主管Vlad…

三个视角解读ChatGPT在教学创新中的应用

第一&#xff0c;我们正处于一个学生使用ChatGPT等AI工具完成作业的时代&#xff0c;传统的教育方法需要适应变化。 教育工作者不应该因为学生利用了先进技术而惩罚他们&#xff0c;相反&#xff0c;应该专注于让学生去挑战超越AI能力范围的任务。这需要我们重新思考教育策略和…

开源vue动态表单组件

一、项目简介 vueelement的动态表单组件&#xff0c;拖拽组件到面板即可实现一个表单 二、实现功能 支持拖拽 支持输入框 支持文本框 支持数字输入框 支持下拉选择器 支持多选框 支持日期控件 支持开关 支持动态表格 支持上传图片 支持上传文件 支持标签 支持ht…