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

这篇具有很好参考价值的文章主要介绍了[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.默认构造函数的应用

  使用情况一:类内初始值

[C++]构造与毁灭:深入探讨C++中四种构造函数与析构函数,C++,数据结构,开发语言,c++,c语言,笔记

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

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

[C++]构造与毁灭:深入探讨C++中四种构造函数与析构函数,C++,数据结构,开发语言,c++,c语言,笔记

      由此报错可以看出,对象数组的创建,必须使用默认的构造函数,在上面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。这证实了在派生类中没有显式定义构造函数时,它将继承基类的默认构造函数,并能够正确初始化基类的成员变量。

注意:

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

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

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

自定义的默认构造函数

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

[C++]构造与毁灭:深入探讨C++中四种构造函数与析构函数,C++,数据结构,开发语言,c++,c语言,笔记

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


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

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

[C++]构造与毁灭:深入探讨C++中四种构造函数与析构函数,C++,数据结构,开发语言,c++,c语言,笔记

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

[C++]构造与毁灭:深入探讨C++中四种构造函数与析构函数,C++,数据结构,开发语言,c++,c语言,笔记

[C++]构造与毁灭:深入探讨C++中四种构造函数与析构函数,C++,数据结构,开发语言,c++,c语言,笔记

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

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

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

[C++]构造与毁灭:深入探讨C++中四种构造函数与析构函数,C++,数据结构,开发语言,c++,c语言,笔记

可以看到,此时通过深拷贝,两个对象的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();

}

[C++]构造与毁灭:深入探讨C++中四种构造函数与析构函数,C++,数据结构,开发语言,c++,c语言,笔记

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++,数据结构,开发语言,c++,c语言,笔记


总结:

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


[C++]构造与毁灭:深入探讨C++中四种构造函数与析构函数,C++,数据结构,开发语言,c++,c语言,笔记

到了这里,关于[C++]构造与毁灭:深入探讨C++中四种构造函数与析构函数的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 构造函数与析构函数

    在上一篇内容中,自定义的类里面是存在成员函数的;而这些成员函数都是我们显式定义出来的,而一般情况下,类中还存在几个默认的成员函数: 构造函数、析构函数 、拷贝构造函数、赋值重载以及取地址重载等,而本节则主要论述前两个函数,构造函数和析构函数! 关

    2024年02月08日
    浏览(41)
  • string类构造函数与析构函数

    构造函数 作用 string s 构造一个空字符串 string s(s1) 生成一个和s1相同的空字符串s string s(s1,5) 将s1[5]以后的部分作为s的初始部分 string s(s1,5,5) 将始于s1[5],长度为5的部分作为s的初始值 string s(cstr) 以C_string类型的cstr作为字符串s的初始值 string s(sctr,char_len) 以C_string类型的cstr的前

    2024年02月08日
    浏览(48)
  • 【C/C++】构造函数与析构函数

    创作不易,本篇文章如果帮助到了你,还请点赞 关注支持一下♡𖥦)!! 主页专栏有更多知识,如有疑问欢迎大家指正讨论,共同进步! 🔥c++系列专栏:C/C++零基础到精通 🔥 给大家跳段街舞感谢支持!ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ c语言内容💖:

    2024年02月11日
    浏览(44)
  • C++中四种不同的强制转换方式的区别

    在C++中,有四种不同的强制转换方式,它们分别是静态转换、动态转换、常量转换和重新解释转换。下面是每种转换的区别:         静态转换是最常用的强制转换方式之一,可以在具有良好定义的类型之间进行转换。它可以处理隐式转换以及非多态类型之间的转换。例

    2024年02月13日
    浏览(52)
  • Roscpp中构造函数与析构函数及在无人机串口通信应用

    构造函数 作用:初始化,在创建对象时会自动调用,有点像Python中的 __init__ 方法 特点:命名为 class命名 ,在public中声明。 析构函数 作用:完成释放前的一些清理工作。 特点:命名为 ~+class命名 ,在public中声明。 调用顺序 栈的顺序,先构造的后析构。 ROScpp中在include文件

    2024年01月21日
    浏览(46)
  • C++奇迹之旅:深入思考拷贝构造函数

    在现实生活中,可能存在一个与你一样的自己,我们称其为双胞胎。 那在创建对象时,可否创建一个与已存在对象一某一样的新对象呢? 拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动

    2024年04月26日
    浏览(38)
  • 深入分析C++对象模型之移动构造函数

    接下来我将持续更新“深度解读《深度探索C++对象模型》”系列,敬请期待,欢迎关注!也可以关注公众号:iShare爱分享,自动获得推文和全部的文章列表。 C++11新标准中最重要的特性之一就是引入了支持对象移动的能力,为了支持移动的操作,新标准引入了一种新的引用类

    2024年04月22日
    浏览(48)
  • 【c++】类和对象(四)深入了解拷贝构造函数

    🔥 个人主页 :Quitecoder 🔥 专栏 :c++笔记仓 朋友们大家好啊,本篇内容带大家深入了解 拷贝构造函数 拷贝构造函数是一种特殊的构造函数,在对象需要以同一类的另一个对象为模板进行初始化时被调用。它的主要用途是初始化一个对象,使其成为另一个对象的副本 我们先

    2024年04月16日
    浏览(69)
  • c++ 学习 之 构造函数的分类和调用类型 深入学习

    构造函数是在C++中用于创建和初始化对象的特殊函数。构造函数可以根据不同的特性和参数进行分类,以下是一些常见的构造函数分类和详细讲解它们的调用方式: 默认构造函数: 默认构造函数是一个特殊的构造函数,它没有参数,也可以没有具体的实现代码。如果没有显

    2024年02月10日
    浏览(40)
  • 深入理解和应用C++ std::shared_ptr别名构造函数

    在现代C++中,智能指针是一个极为重要的工具,尤其std::shared_ptr以其自动内存管理、引用计数和多线程安全性等特性深受开发者喜爱。其中一个不太常用但功能强大的构造方式是 别名构造函数 ,它允许我们创建一个共享相同底层对象但是指向其内部不同数据成员或子对象的

    2024年01月16日
    浏览(45)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包