C++从汇编来分析C++构造函数和析构函数

C++构造函数和析构函数

我们知道C++中有类和对象概念,对对象的操作我们一般通过函数实现,比如定义一个Dog类,实例化一个叫阿黄的小狗,狗可以执行跑步动作,可以旺旺的叫,这些功能都是通过函数来实现的。而构造函数是类的特殊函数,创建对象时,会自动调用构造函数(是先给对象分配空间,然后调用构造函数,而不是调用构造函数去给对象分配空间,下面会对此分析),我们可以利用构造函数对成员变量进行初始化操作。看个例子:

#include<iostream>
using namespace std;

class Dog {
public:
	Dog() {}
	~Dog() {}

	void run(void) {
		cout << "dog is running!"<< endl;
	}

private:
};

int main(){
	Dog dog;
	dog.run();
	cout << sizeof(Dog) << endl;
	return 0;
}

类的成员函数在编译时就确定了,位于代码段,只有一份,并且不占用对象的空间,有意思的是为什么成员函数只有一份,它却可以访问到不同对象的成员变量?答案就是this指针, 当通过对象去调用函数时,这个唯一的函数的this指针指向了这个对象。上面代码的输出的类大小是1字节,这是为什么呢?虽然这个类没有成员变量,但是对象被创建出来肯定需要占用空间,不然怎么表示这个对象的存在,而且this指针也需要是这个对象空间的地址。

1. 构造函数

经过上面的分析,空类占用一个1个字节大小,非空类的大小就是成员变量的大小(需要注意的是和c语言struct一样,存在字节对齐的情况,比如int a; char b;这时的大小是8字节,而不是4+1=5字节)。构造函数的作用定义对象时调用构造函数对其初始化,上述代码定义对象是吧对象创建在栈中,出了生命周期后会自动释放,采用new的方式创建对象是在堆中,需要程序员手动释放。

1.1 默认构造函数和重写构造函数

类中如果我们不写构造函数,那么就存在默认的构造函数,默认的构造函数等效于上例Dog() {}的写法,函数体里没有内容,既然没有内容,那么是怎么构建出对象来的?笔者认为是这样的:构造函数只是对成员变量进行初始化操作,给变量分配内存空间在定义对象时就分配好了,如果构造函数没有内容,成员变量的值就是不确定的垃圾值,这里我使用IAR对c++代码编译成汇编,看汇编中是对构造函数进行了怎样的处理。

1.1.1 构造函数为空时(或者是默认构造函数)

C++代码:

class Test {
public:
	Test() {
//        a = 1;
//        b = 2;
//        c = 3;
//        d = 4;
//        e = 5;
    }

public:
	int a;
    int b;
    int c;
    int d;
    int e;
};

int test() {
    Test t;
    t.a = 1;
    t.b = 2;
    t.c = 3;
    t.d = 4;
    t.e = 5;

	return 0;
}

编译后的汇编代码:
在这里插入图片描述
从汇编代码中我们看到并没有调用构造函数,因为构造函数为空,编译器编译后并不存在构造函数的内容,而是直接将堆栈指针减去了20,5个int型变量占20个字节。

1.1.2 构造函数中对变量初始化
  • C++代码:
class Test {
public:
	Test() {
        a = 1;
        b = 2;
        c = 3;
        d = 4;
        e = 5;
    }

public:
	int a;
    int b;
    int c;
    int d;
    int e;
};

int test() {
    Test t;
//    t.a = 1;
//    t.b = 2;
//    t.c = 3;
//    t.d = 4;
//    t.e = 5;

	return 0;
}
  • 生成的汇编代码:
    在这里插入图片描述
    从上图我们可知,给成员变量分配内存空间后,调用构造函数中对变量进行赋值。

1.2 类比C语言中的结构体

C语言中定义一个结构体变量,也是给变量分配内存空间,与C++中的struct和class类似,区别在于C++中可以自己写构造函数对变量初始化,省去了C语言中,需要手动调用初始化函数的麻烦。

2. 析构函数

如果对象定义在栈中,出了对象的生命周期,对象会自动释放,在释放之前会调用析构函数。如果析构函数为默认的析构函数,相当于没有调用。

  • C++:
#include<iostream>

class Test {
public:
	Test() {
        a = 1;
        b = 2;
        c = 3;
        d = 4;
        e = 5;
    }
    ~Test() {
        a = 0;
        b = 0;
        c = 0;        
    }

public:
	int a;
    int b;
    int c;
    int d;
    int e;
};

int test() {
    Test t;
//    t.a = 1;
//    t.b = 2;
//    t.c = 3;
//    t.d = 4;
//    t.e = 5;

	return 0;
}
  • 汇编代码
    在这里插入图片描述
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值