c++构造函数和析构函数

本文详细介绍了C++中的构造函数、析构函数和拷贝构造函数。构造函数用于对象初始化,无参构造函数默认生成,可重载。析构函数负责清理对象,与构造函数作用相反。拷贝构造函数用于创建新对象并复制已有对象属性,分为浅拷贝和深拷贝。文章通过实例代码阐述了这些概念及其应用场景。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

c++构造函数和析构函数

构造函数

💬 什么是构造函数
🔑构造函数就是一个特殊的类成员函数,函数名和类名相同,它在创建类类型对象时会由编译器自动调用

函数名()
{

}
函数名(参数1,参数2)
{

}

💬构造函数有什么作用
🔑构造函数主要用来在创建对象时初始化对象, 即为对象成员变量赋初始值,总与new运算符一起使用在创建对象的语句中。特别的一个类可以有多个构造函数 ,可根据其参数个数的不同或参数类型的不同来区分它们 (即构造函数的重载)。

🎯构造函数的特点
① 函数名和类名相同
② 没有返回值
③ 如果不写构造函数,任何类中都存在一个默认的构造函数

默认的构造函数是无参的。 		
当我们自己写了构造函数,默认的构造函数就不存在

④ 构造函数在构造对象的时候调用
⑤ delete可以用来删掉默认的函数
⑥ 指定使用默认的无参构造函数,用default说明
⑦ 允许构造函数调用另一个构造函数,只是要用初始化参数列表的写法
⑧ 初始化参数列表 : 只有构造函数有
构造函数名(参数1,参数2,…):成员1(参数1),成员2(参数2),…{}
⑨避免形参名和数据成员名相同的导致问题

🎯构造函数代码分析

#include<iostream>
#include<string>
using namespace std;

class Student
{
public:
	Student()
	{
		_name = "莫九梦";
		_age = 20;
		_id = 111;
	}
	Student(string name, int age, int id)
	{
		_name = name;
		_age = age;
		_id = id;
	}
	void print()
	{
		cout << _name << " " << _age << " " << _id << endl;
	}
protected:
	string _name;
	int _age;
	int _id;
};

int main()
{
	Student student;//触发构造,调用无参构造函数
	student.print();
	Student student1("梦", 21, 2020);//触发构造,调用有参构造函数
	student1.print();
	return 0;
}

🎡运行结果
在这里插入图片描述
🎯默认构造函数分三种:

①我们不写,编译器就自己生成的构造函数
② 无参的构造函数
③ 全缺省的构造函数(效果和无参构造函数一样)

😈注意:
构造函数是特殊的类成员函数,不能用student.Student();的方式调用

Student student;
student.Student();
//错误的调用方法

通过无参构造函数创建对象,对象后面不能跟括号,否则就成了函数声明。

#include<iostream>
#include<string>
using namespace std;

class Student
{
public:
	Student() {}
	Student(string name)
	{
		_name = name;
	}
	void print()
	{
		cout << _name << endl;
	}
protected:
	string _name;
};

int main()
{
	Student student();
	// 这个对象实际上没有被定义出来,这里会报错。
	//调用无参构造函数不能带括号,就用不带括号的来定义就行了
	//Student student;
	return 0;
}

如果你没有定义构造函数(类中未显式定义),C++ 编译器就会自动生成一个默认的无参构造函数。当然,如果你自己定义了,编译器就不会帮你生成

#include<iostream>
using namespace std;

class Data
{
public:
	void print()
	{
		cout << data << endl;
	}
protected:
	int data;
};

int main()
{
	//没有定义构造函数,对象也可以创建成功
	Data data;//这里调用的是默认的无参构造函数
	data.print();
	return 0;
}

在这里插入图片描述
定义了无参构造函数不能同时构造全缺省构造函数

//错误代码
#include<iostream>
#include<string>
using namespace std;

class Student
{
public:
	Student()
	{
		_name="莫九梦";
	}
	Student(string name="莫九梦")
	{
		_name = name;
	}
	void print()
	{
		cout << _name << endl;
	}
protected:
	string _name;
};

int main()
{
	Student student;
	
	return 0;
}

在这里插入图片描述

析构函数

既然知道了怎么初始化对象,那么如何清理对象呢?这里就要用到析构函数
🔥先来讲讲析构函数的概念
析构函数和构造函数一样,也是特殊的类成员函数,作用吗?和构造函数相反,析构函数主要是用来清理对象的

这里要注意了!!!
构造函数主要任务是初始化对象,而不是创建对象
析构函数主要是清理对象,而不是销毁对象

🎯析构函数的特点
① 析构函数名是在类名前面加个~
② 没有参数也没有返回值(因为没有参数,所以也不会构成重载问题)
③ 释放数据成员new的内存

在对象死亡前自动调用   
通常如果数据成员没有做new操作,就可以不写析构函数

④ 不写析构函数,存在一个默认的析构函数
⑤ 一个类的析构函数有且仅有一个(如果不写系统会默认生成一个析构函数)

🎡这里我们写个代码看看析构函数的自动调用

#include<iostream>
#include<string>
using namespace std;


class Student
{
public:
	Student(string name = "莫九梦");
	~Student();
	void print()
	{
		cout << _name << endl;
	}
private:
	string _name;
};

Student::Student(string name)
{
	_name = name;
}

Student::~Student()
{
	cout << "调用了析构函数" << endl;
}

int main()
{
	Student student;
	Student student1("九梦");
	return 0;
}

在这里插入图片描述

拷贝构造函数

① 拷贝构造函数也是构造函数
② 拷贝构造函数参数是固定的:对对象的引用
③ 拷贝构造函数不写会存在一个默认的拷贝构造函数
④ 拷贝构造函数作用: 通过一个对象产生另一个对象

关键点:一定是有一个新的对象产生

👀拷贝构造函数的概念

拷贝构造函数,又称复制构造函数,是一种特殊的构造函数,它由编译器调用来完成一些基于同一类的其他对象的构建及初始化。(它是一个特殊的成员函数)

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

👀拷贝构造函数的特点
① 拷贝构造函数是构造函数的一个重载形式。函数名和类名相同,没有返回值。
② 拷贝构造函数的参数只有一个,并且必须要使用引用传参

💬代码分析

#include <iostream>
#include <string>
using namespace std;
class Student
{
public:
	Student() {}
	Student(string name, int age)
	{
		_name = name;
		_age = age;
	}

	//拷贝构造函数
	//通过传入对象属性确定创建对象的属性
	Student(Student& object)
	{
		_name = object._name;
		_age = object._age;
		cout << "调用自己写的拷贝构造函数" << endl;
	}

	void print()
	{
		cout << _name << "\t" << _age << endl;
	}
protected:
	string _name;
	int _age{};
};

//函数传参也可以隐式调用
void print(Student object)		//Student object=girl,产生了新的对象object
{
	object.print();
}
//C++中传参能用引用就用,效率搞
void printData(Student& object) //引用就是别名,没有产生新的对象
{
	object.print();
}

int main()
{
	Student student("莫九梦", 20);		//产生一个对象
	Student beauty = student;			//隐式调用拷贝构造函数
	Student girl(beauty);				//显式调用拷贝构造函数
	print(girl);						//调用拷贝构造函数
	printData(girl);

	
	Student boy;			//对象先有了,才赋值,就不调用拷贝
	boy = girl;				//不调用拷贝构造函数-->默认重载=运算符(后续会讲)

	return 0;
}

在这里插入图片描述

深浅拷贝问题

浅拷贝

  • 没有在拷贝构造函数中给数据成员做new操作
  • 默认拷贝构造函数都是浅拷贝

🌞深拷贝

  • 在拷贝构造函数中做了new操作

代码分析

#include <iostream>
#include <cstring>
using namespace std;
class MM 
{
public:
	MM() {}
	MM(const char* str) 
	{
		int length = strlen(str) + 1;
		name = new char[length];
		strcpy_s(name, length, str);
	}
	~MM() 
	{
		if (name != nullptr) 
		{
			delete[] name;
			name = nullptr;
		}
	}
private:
	char* name;
};
int main() 
{
	{
		MM girl("girl");
		MM mm = girl;			//调用拷贝构造函数
	}
	return 0;
}

在这里插入图片描述

🚗解决方案
通过深拷贝来解决

#include <iostream>
#include <cstring>
using namespace std;
class MM 
{
public:
	MM() {}
	MM(const char* str) 
	{
		int length = strlen(str) + 1;
		name = new char[length];
		strcpy_s(name, length, str);
	}
	MM(MM& object) 
	{
		//name = object.name;
		int length = strlen(object.name) + 1;
		name = new char[length];
		strcpy_s(name, length, object.name);
	}
	~MM() 
	{
		if (name != nullptr) 
		{
			delete[] name;
			name = nullptr;
		}
	}
private:
	char* name;
};
int main() 
{
	{
		MM girl("girl");
		MM mm = girl;			//调用拷贝构造函数
	}
	return 0;
}

综上: 一旦类中有指针,做了内存申请,并且要对对象做拷贝操作,就必须使用深拷贝

构造和析构顺序问题

  • 一般情况构造顺序和析构是相反的
  • 静态的和全局的是最后释放的
  • delete 立刻调用析构函数
#include <iostream>
#include <string>
using namespace std;
class Test
{
public:
	Test(string data = "A")
	{
		m_data = data;
		cout << m_data;
	}
	~Test()
	{
		cout << m_data;
	}
protected:
	string m_data;
};

int main()
{
	{
		Test t1;					//A
		static Test t2("B");		//B
		Test array[3];				//AAA   //数组就是多个无参
		Test* p = new Test("C");	//C
		delete p;					//C
		p = nullptr;
	}
	//先构造函数输出ABAAAC,再析构函数输出CAAAAB
	//说明构造函数和析构函数的调用顺序是相反的
	return 0;
}

在这里插入图片描述

### 构造函数析构函数的概念 在C++中,构造函数是一个特殊的成员函数,在创建对象时自动调用,用于初始化对象的状态[^3]。它具有与类名相同的名称,并且返回任何值(甚至需要指定 `void`)。如果未显式定义构造函数,则编译器会提供一个默认构造函数。 相反,析构函数是在对象生命周期结束时被调用的一个特殊成员函数,其主要职责是释放由该对象分配的资源,例如动态内存或其他外部资源[^4]。它的命名方式也是与类同名,但在前面加上波浪号 (`~`)。 ### 使用方法 #### 构造函数 构造函数可以有多种形式,包括无参构造函数、带参数的构造函数以及复制构造函数。下面展示了一个简单的例子: ```cpp class MyClass { public: int value; // 默认构造函数 MyClass() : value(0) {} // 带参数的构造函数 MyClass(int val) : value(val) {} }; ``` 上述代码展示了如何通过同的构造函数来设置对象初始状态的方法[^5]。 #### 析构函数 析构函数通常用来清理工作,比如关闭文件句柄或者删除动态分配的对象。这里有一个使用指针的例子: ```cpp class ResourceHandler { private: char* data; public: ResourceHandler(size_t size) { data = new char[size]; } ~ResourceHandler() { delete[] data; // 清理动态分配的内存 } }; ``` 在这个实例里,当 `ResourceHandler` 对象销毁的时候,析构函数会被触发并执行必要的清除操作以防止内存泄漏等问题发生[^6]。 ### 主要区别 | 特性 | 构造函数 | 析构函数 | |-----------------|---------------------------------------|-----------------------------------| | **调用时机** | 创建新对象时 | 销毁现有对象前 | | **功能目的** | 初始化对象 | 执行清理任务 | | **重载支持** | 支持 | 支持 | | **返回类型** | 无 | 无 | 需要注意的是,尽管两者都属于特殊类型的成员函数,但是它们的行为模式完全同——前者负责建立有效的运行环境而后者则专注于解除这些绑定关系以便回收利用系统资源[^7]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值