构造函数、析构函数及深浅拷贝

平时我们在书写C++代码时,很少书写其构造函数以及析构函数,因为它们都是有默认实现的。而且也不需要我们手动调用,是系统自动调用的。但是了解构造函数、析构函数可以帮助我们更好的了解到深浅拷贝。

1.什么是构造函数 与 析构函数?

构造函数的主要作用在于创建对象时可以为对象的成员属性赋初值;

析构函数主要用于对象销毁前执行一些清理的工作。

2.构造函数的分类

构造函数可以分为无参构造、有参构造、以及拷贝构造

3.构造函数的语法

class Person {

public:
	//无参构造函数
	Person() {
	
	}
	//有参构造函数
	Person(int age) {
		m_Age = age;
	}
	//拷贝构造函数
	Person(Person& p) {
		m_Age = p.m_Age;
	}
	//析构函数
	~Person() {
	
	}
private:
	int m_Age;
};

其中需要注意的是,拷贝构造函数的参数 一定是引用传递。因为如果是值传递,值传递的本质就是调用拷贝构造函数,会导致无限递归的调用拷贝构造函数。加const 是为了防止误操作修改本体对象的内容。主函数调用构造函数时,一般使用括号法,语法比较简单,不再赘述。但又几点需要注意:1)不要用括号法调用无参构造函数。 例如Person p(); 编译器会认为这是一个函数声明,直接Person p;实例化一个对象即可。2)不要用拷贝构造函数初始化匿名对象。例如Person(p2); 编译器会认为 这行代码为:Person p2; 为一个对象实例化的操作。

4.构造函数和析构函数的默认实现

如果不手动书写构造函数和析构函数,那么编译器会为我们写的类增加3个函数

1)默认构造函数 --无参、空实现

2)默认析构函数 --无参、空实现

3)默认拷贝构造函数 -- 简单的值拷贝操作(浅拷贝)

如果用户自己定义了有参构造函数,编译器不会提供默认构造函数,但会提供拷贝构造函数(浅拷贝);

如果用户自己定义了拷贝构造函数,那么编译器不会提供任何默认构造函数。

5.浅拷贝 出现的原因以及解决办法(深拷贝)

我们知道构造函数的作用是会为对象属性赋初值;析构函数会再释放对象前,做清理的工作。

下面假设,我们创建一个类,他的属性中有指针,并且指针指向动态分配的内存空间,析构函数做了动态内存释放的处理。

class Person{
public:
	Person(char* name,int age){
		pName = (char*)malloc(strlen(name) + 1);
		strcpy(pName,name);
		mAge = age;
	}
	//增加拷贝构造函数
	Person(const Person& person){
        // 重新在堆区内创建一块内存
		pName = (char*)malloc(strlen(person.pName) + 1);
		strcpy(pName, person.pName);
		mAge = person.mAge;
	}
	~Person(){
		if (pName != NULL){
			free(pName);
		}
	}
private:
	char* pName;
	int mAge;
};

void test(){
	Person p1("Edward",30);
	//用对象p1初始化对象p2,调用c++提供的默认拷贝构造函数
	Person p2 = p1;
}

如果调用默认拷贝构造函数,会出现堆区开辟的内存重复释放的错误;(浅拷贝问题)

解决方式是自己写一个拷贝构造函数,再到堆区创建一块新的内存。(深拷贝)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: "undefined reference to constructor/destructor" 是编译器给出的一种错误信息,通常是由于没有找到该类的构造函数析构函数的定义导致的。这通常发生在以下情况: 1. 类的头文件未包含,导致编译器无法识别该类。 2. 构造函数/析构函数的定义与声明不一致,导致编译器无法识别。 3. 在类外定义了构造函数/析构函数,但是未在头文件中声明,导致编译器无法识别。 解决方案: 1. 检查包含的头文件是否正确,确保类的头文件已包含。 2. 检查构造函数/析构函数的定义与声明是否一致,修正不一致的地方。 3. 在类的头文件中声明构造函数/析构函数。 ### 回答2: 当编译器发现一个"undefined reference to"的错误时,通常意味着代码中引用但未定义的函数。而在构造函数析构函数的情况下,这可能是因为以下原因导致的错误: 1. 构造函数析构函数的声明与定义不匹配:代码中声明了一个构造函数析构函数,但没有提供与该声明匹配的实现。这可能是因为构造函数析构函数的名称、参数列表或返回类型与定义不匹配,导致编译器无法找到相应的定义。要解决这个问题,我们需要确保所有构造函数析构函数的声明和定义是匹配的,包括名称、参数列表和返回类型。 2. 构造函数析构函数的定义缺失:在某些情况下,可能会在代码中声明构造函数析构函数,但没有提供相应的定义。这种情况下,编译器在链接阶段无法找到构造函数析构函数的定义,从而报错。要解决这个问题,我们需要确保所有构造函数析构函数都有相应的实现,即提供与其声明匹配的定义。 综上所述,当出现"undefined reference to 构造函数 析构函数"的错误时,通常是因为构造函数析构函数的声明与定义不匹配,或者缺少相应的定义。通过检查代码并确保所有构造函数析构函数的声明和定义匹配,我们可以解决这个错误。 ### 回答3: 在编程中,当我们在使用一个类的构造函数析构函数时,有时候会遇到"undefined reference to 构造函数 析构函数"的错误。这个错误发生在链接阶段,通常是由于以下几种情况导致的: 1. 构造函数析构函数未定义:编译器找不到对应的构造函数析构函数的定义。要解决该问题,需要在实现文件中提供对应的定义,确保在使用时能够找到定义的实现。 2. 构造函数析构函数的定义与声明不匹配:在类的声明中声明并未提供参数列表,但在定义时提供了参数列表,或者反之。要解决该问题,需要检查类的声明和定义是否一致,确保参数列表的匹配。 3. 构造函数析构函数的访问权限不正确:如果构造函数析构函数在类的声明中被声明为私有的,而在调用时却被当作公有的进行访问,就会出现该错误。要解决该问题,需要检查类的访问权限,确保在使用时能够正确访问。 4. 构造函数析构函数的定义未被编译到可执行文件中:检查是否在编译过程中将实现文件添加到了编译器的编译选项中,确保构造函数析构函数的实现能够被编译和链接到可执行文件中。 总结来说,当编译器在链接阶段遇到"undefined reference to 构造函数 析构函数"的错误时,需要检查构造函数析构函数的定义和声明是否一致,确保可执行文件中能正确找到这些函数的实现。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值