C++学习笔记-构造函数和析构函数
构造函数和析构函数是C++的重要组成部分,了解构造函数和析构函数有助于深入了解C++
构造函数
构造函数产生的原因
在C++中,有时候需要在对象创建的时候初始化数据,如果采用普通函数的话,每次初始化都要调用函数,显得麻烦,另外,如果忘记初始化,那么其结果是未知的。
在C++中使用构造函数来完成初始化,使得对象在生成的时候就完成初始化动作
构造函数定义
- C++中的类可以定义与类名相同的特殊成员函数,这种与类名相同的成员函数叫做构造函数
- 构造函数在定义时可以有参数
- 没有任何返回类型的声明
class Test
{
public:
//构造函数 无参构造函数 默认构造函数
Test()
{
a = 10;
}
//带参数的构造函数
Test(int mya)
{
a = mya;
}
//赋值构造函数 copy构造函数
Test(const Test & obj)
{
;
}
protected:
private:
int a;
};
构造函数的调用
- 自动调用:一般情况下C++编译器会自动调用构造函数
- 手动调用:在一些情况下则需要手工调用构造函数
//无参构造函数调用,适合C++默认的构造函数
Test t0;
//有参构造函数掉用:3种
Test t1(10); //自动调用
Test t2 = 11; //自动调用
Test t3 = Test(12); //手动调用
//copy构造函数调用:4种
Test t4;
//第一种:
Test t5 = t4;//定义并初始化,用t4初始化t5-->t5 = t4不同,后者属于浅拷贝
//第二种
Test t6(t4);
//第三种
//如果存在函数void func(Test p){},那么在类做函数参数的时候会调用copy构造函数
Test t7(10);
func(t7);
//第四种
//在定义返回值为类的函数时,会调用copy构造函数,此时生成一个匿名对象
Test func()
{
Test t8(10);
return t8;
}
//这个函数引出匿名对象的生命周期问题
Test t9;
t9 = func();//这种方法会有三次构造三次析构,属于浅拷贝
Test t9 = func();//这种方法直接将匿名对象转化为t9,只经历两次构造和两次析构
特殊的构造函数
- 无参构造函数
当类中没有定义构造函数时,编译器默认提供一个无参构造函数,并且其函数体为空 - 拷贝构造函数
当类中没有定义拷贝构造函数时,编译器默认提供一个拷贝构造函数,简单的进行成员变量的值复制
构造函数的规则和理解
- 当类中没有定义任何一个构造函数时,c++编译器会提供无参构造函数和拷贝构造函数
- 当类中定义了任意的非拷贝构造函数,c++编译器不会提供无参构造函数
- 当类中定义了拷贝构造函数时,c++编译器不会提供无参数构造函数
- 默认拷贝构造函数成员变量简单赋值
- 构造函数是C++中用于初始化对象状态的特殊函数
- 构造函数在对象创建时自动被调用
- 构造函数和普通成员函数都遵循重载规则
- 拷贝构造函数是对象正确初始化的重要保证
- 必要的时候,必须手工编写拷贝构造函数
构造函数的初始化列表
产生的原因
由于在多个类中存在包含关系,导致在包含其他类的类中初始化不明确
此时会造成没有合适的默认构造函数
如果有一个类成员,它本身是一个类或者是一个结构,而且这个成员它只有一个带参数的构造函数,没有默认构造函数。这时要对这个类成员进行初始化,就必须调用这个类成员的带参数的构造函数,如果没有初始化列表,那么他将无法完成第一步,就会报错
使用
Constructor::Contructor() : m1(v1), m2(v1,v2), m3(v3)
{
// some other assignment operation
}
初始化:被初始化的对象正在创建
赋值:被赋值的对象已经存在
成员变量的初始化顺序与声明的顺序相关,与在初始化列表中的顺序无关
初始化列表先于构造函数的函数体执行
#include "iostream"
using namespace std;
class ABC
{
public:
ABC(int a, int b, int c)
{
this->a = a;
this->b = b;
this->c = c;
printf("a:%d,b:%d,c:%d \n", a, b, c);
printf("ABC construct ..\n");
}
~ABC()
{
printf("a:%d,b:%d,c:%d \n", a, b, c);
printf("~ABC() ..\n");
}
protected:
private:
int a;
int b;
int c;
};
class MyD
{
public:
MyD():abc1(1,2,3),abc2(4,5,6),m(100)
{
cout<<"MyD()"<<endl;
}
~MyD()
{
cout<<"~MyD()"<<endl;
}
protected:
private:
ABC abc1;
ABC abc2;
const int m;
};
int run()
{
MyD myD;
return 0;
}
int main()
{
run();
system("pause");
return 0;
}
#include "iostream"
using namespace std;
class ABCD
{
public:
ABCD(int a, int b, int c)
{
this->a = a;
this->b = b;
this->c = c;
printf("ABCD() construct, a:%d,b:%d,c:%d \n", this->a, this->b, this->c);
}
~ABCD()
{
printf("~ABCD() construct,a:%d,b:%d,c:%d \n", this->a, this->b, this->c);
}
int getA()
{
return this->a;
}
protected:
private:
int a;
int b;
int c;
};
class MyE
{
public:
MyE():abcd1(1,2,3),abcd2(4,5,6),m(100)
{
cout<<"MyD()"<<endl;
}
~MyE()
{
cout<<"~MyD()"<<endl;
}
MyE(const MyE & obj):abcd1(7,8,9),abcd2(10,11,12),m(100)
{
printf("MyD(const MyD & obj)\n");
}
protected:
//private:
public:
ABCD abcd1;
ABCD abcd2;
const int m;
};
int doThing(MyE mye1)
{
printf("doThing() mye1.abc1.a:%d \n", mye1.abcd1.getA());
return 0;
}
int run()
{
MyE myE;
doThing(myE);
return 0;
}
int main()
{
run();
system("pause");
return 0;
}
析构函数
析构函数产生的原因
构造函数在对象被创建时候调用,而析构函数则相反,是在对象被销毁时候调用,用于清理内存或者变量
C++中的类可以定义一个特殊的成员函数清理对象,这个特殊的成员函数叫做析构函数
析构函数定义
~ClassName()
- 析构函数没有参数也没有任何返回类型的声明
- 析构函数在对象销毁时自动被调用
- 析构函数调用机制:C++编译器自动调用
总结
当类中有成员变量是其它类的对象时
首先调用成员变量的构造函数
调用顺序与声明顺序相同
之后调用自身类的构造函数
析构函数的调用秩序与对应的构造函数调用秩序相反
class Test
{
public:
Test()
{
cout<<"构造函数自动被调用"<<endl;
}
~Test()
{
cout<<"析构函数自动被调用"<<endl;
}
protected:
private:
};
···
C++学习笔记-构造函数和析构函数的更多相关文章
- 学习C++ -> 构造函数与析构函数
学习C++ -> 构造函数与析构函数 一.构造函数的介绍 1. 构造函数的作用 构造函数主要用来在创建对象时完成对对象属性的一些初始化等操作, 当创建对象时, 对象会自动调用 ...
- C++学习之构造函数和析构函数及指针
C++的构造函数在创建对象时调用,分配内存空间,多少个对象(对象数组)就调用几次构造函数:析构函数在调用结束时调用(可以添加一些最后的处理)以释放内存给其它来用.对于同类型同生命期的对象,先创建的对象 ...
- 鸡啄米:C++编程之十四学习之构造函数和析构函数
1. 本人学习鸡啄米课程的笔记记录,用来记录学习的历程和进度 2. 构造函数 我们在声明一个变量时,如果对它进行了初始化,那么在为此变量分配内存空间时还会向内存单元中写入变量的初始化.声明对象有相似的 ...
- C++笔记----构造函数与析构函数(二)
构造函数是一种特殊的成员函数.在创建对象的时候自动调用,对对象的数据成员进行初始化. (1)栈区中创建的对象,在生存期结束的时候会自动调用析构函数. (2) 在堆上创建的对象,要由程序员显示调用del ...
- C++笔记----构造函数与析构函数(三)
1.构造函数初始化列表 推荐在构造函数初始化列表中进行初始化 构造函数的执行分为两个阶段:初始化段. 普通计算段 2.对象成员及其初始化 #include<iostream> using ...
- c++学习笔记——构造函数
构造函数定义:每个类都分别定义了它的对象被初始化的方式,类通过一个或几个特殊的成员函数来控制其对象的初始化过程,这些函数叫做构造函数. 需要注意的几点: 1:构造函数不能被声明为const的,当我们创 ...
- JavaScript学习笔记-构造函数
什么是构造函数 简单说构造函数是类函数,函数名与类名完全相同,且无返回值.构造函数是类的一个特殊成员函数. JavaScript构造函数 * 在JavaScript的世界里没有类的概念,JavaScr ...
- JS学习笔记-构造函数篇
创建实例 funtion Fn (){ var num = 10; this.x = 100; this.getX = function(){ cons ...
- 《C++ Primer Plus》10.3 类的构造函数和析构函数 学习笔记
10.3.1 声明和定义构造函数构造函数原型:// constructor prototype with some default argumentsStock(const string &c ...
随机推荐
- 【计算机-虚拟wifi】Win7虚拟wifi设置
虚拟Wifi,可以让电脑变成无线路由器,实现共享上网. 设置步骤 1.以管理员身份运行:cmd.exe 2.启动并设置虚拟网卡: 命令窗口中输入: netsh wlan set host ...
- [人物存档]【AI少女】【捏脸数据】魔幻世界
人物+衣物.zip
- Ubuntu安装jdk10
一:去官网下载jdk,和jre 因为jdk10之后jdk和jre是分开的 jdk下载 jre下载 二:解压缩,并放到指定目录 # 创建目录 sudo mkdir /usr/lib/java ...
- 两个线程,一个线程打印1~52,另一个线程打印字母A-Z,打印顺序为12A34B56C……5152Z
使用wait,notify实现 public class Test { public synchronized void a() { for (int i = 1; i <= 52; i++) ...
- AcWing:146. 序列(小根堆 + 数学归纳 + 贪心)
给定m个序列,每个包含n个非负整数. 现在我们可以从每个序列中选择一个数字以形成具有m个整数的序列. 很明显,我们一共可以得到nmnm个这种序列, 然后我们可以计算每个序列中的数字之和,并得到nmnm ...
- [题解] [SDOI2010] 古代猪文
题面 题解 题目所求即为 \[ G ^ {\sum_{d | n}C_{n}^{d}} \bmod {999911659} \] 考虑到有这样一个式子 \[ a ^ b \equiv a ^ {b \ ...
- C++入门经典-例8.2-构造函数的访问顺序
1:父类和子类中都有构造函数和析构函数,那么子类对象在创建时是父类先进行构造还是子类先进行构造?同样,在子类对象释放时,是父类先进行释放,还是子类先进行释放?这都是有先后顺序的.答案是当从父类派生一个 ...
- synchronized三种使用方式,及锁的类型验证
Synchronized常用三种使用方式 1.修饰普通方法:锁对象即为当前对象 2.修饰静态方法:锁对象为当前Class对象 3.修饰代码块:锁对象为synchronized紧接着的小括号内的对象 一 ...
- leetcode-hard-array-287. Find the Duplicate Number
mycode 77.79% class Solution(object): def findDuplicate(self, nums): """ :type nums ...
- Struts2与Servlet API的解耦访问