c++中的对象的构造顺序与析构顺序
问题一
当程序中存在多个对象时,如何确定这些对象的析构顺序?
一.单个函数创建时构造函数的调用顺序
1.调用父类的构造过程
2.调用成员变量的构造函数(调用顺序与声明顺序相同)
3.调用类自身的构造函数
而析构函数与对应构造函数的调用顺序相反!多个对象析构时析构顺序与构造顺序想反。
下面分析一段简单的代码:
#include <stdio.h>
class Member
{
const char* ms;
public:
Member(const char* s)
{
printf("Member(const char* s): %s\n", s);
ms = s;
}
~Member()
{
printf("~Member(): %s\n", ms);
}
};
class Test
{
Member mA;
Member mB;
public:
Test() : mB("mB"), mA("mA")
{
printf("Test()\n");
}
~Test()
{
printf("~Test()\n");
}
};
Member gA("gA");
int main()
{
Test t;
return 0;
}
首先代码中没有父类,那么调用成员变量的构造函数,调用的顺序要与声明的顺序相同,看代码知先声明的是全局变量Member gA(“gA”),然后是局部变量:Member mA; Member mB,注意mA和mB这两个变量得顺序不要被 Test() : mB(“mB”), mA(“mA”)这里的定义顺序搞混了,我们的构造顺序是声明的顺序,而不是定义的顺序,那里的定义的顺序是为了给我们造成混淆的,定义的顺序可以随便改变!接着就是调用类自身的构造函数,Test(),它会执行printf(“Test()\n”);
好了,到这一步,说明构造函数调用完成。调用的顺序为:gA,mA,mB,Test().
而析构函数与构造函数的调用顺序相反,所以析构函数的调用顺序为:~Test(), mB, mA, gA.
对于栈对象和全局对象,类似于入栈与出栈的顺序,最后构的对象被最先析构!!
堆对象的析构发生在使用delete的时候,与delete的使用顺序相关!!
问题二
const 关键字能否修饰类的对象?如果能,有什么特性?
我们知道,在c++中,const可以修饰一个只读变量,也可以修饰一个真正意义上的常量。那么它能否修饰类的对象呢?我们知道类只不过是由struct演变而来的一种用户自定义的数据类型,从某种意义上来讲,它也是一个变量,既然是变量,那么能不能用const修饰它呢?
答案是肯定的!
1.const关键字能够修饰对象
2.const修饰的对象为只读对象
3.只读对象的成员变量不允许被改变
3.只读对象是编译阶段的概念,运行时无效
下面介绍一下C++中const成员函数的定义:
Type ClassName::function(Type p) const
类中的函数声明与实际的函数定义都必须带有const关键字。文字太多不如直接上代码:
#include <stdio.h>
class Test
{
int mi;
public:
Test(int i);
Test(const Test& t);
int getMi();
};
Test::Test(int i)
{
mi = i;
}
Test::Test(const Test& t)
{
mi = t.getMi(); //能否编译通过?
}
int Test::getMi()
{
return mi;
}
int main()
{
const Test t(1);
t.mi = 100; //能否编译通过?
printf("t.getMi() = %d\n",t.getMi()); //能否编译通过?如何才能编译通过?
return 0;
}
我把程序放到linux中进行编译,很显然编译不通过,显示的错误有哪些呢?
test.cpp: In copy constructor ‘Test::Test(const Test&)’:
test.cpp:19: error: passing ‘const Test’ as ‘this’ argument of ‘int Test::getMi()’ discards qualifiers
test.cpp: In function ‘int main()’:
test.cpp:5: error: ‘int Test::mi’ is private
test.cpp:31: error: within this context
test.cpp:31: error: assignment of data-member ‘Test::mi’ in read-only structure
test.cpp:33: error: passing ‘const Test’ as ‘this’ argument of ‘int Test::getMi()’ discards qualifiers
首先mi = t.getMi();
无法编译通过,因为Test::Test(const Test& t)中的的参数为const的引用,const成员函数只能调用const成员函数。
然后t.mi = 100;
编译不通过,因为int Test::mi’ is private,并且‘Test::mi’ in read-only structure,因为我们定义的是const成员函数const Test t(1);这才是我们想说的真正原因。
其次printf("t.getMi() = %d\n",t.getMi());
编译不通过 ,因为成员t在上面被定义的是const类型。 那么,该如何让它编译通过呢?在int getMi();
函数后面加上const变为
int getMi()const;
在int Test::getMi()
后面加上const变为
`int Test::getMi()const`
那么这条语句 printf("t.getMi() = %d\n",t.getMi());
就可以编译通过了!!!
此时如果在函数int Test::getMi()
中加入 mi = 2;
则编译又不会通过了,因为该函数已经被定义为const类型,不能改写成员变量的值了!!!
由以上代码的实际试验得出具体结论如下:
C++中const成员函数的特性:
- const const对象只能调用const成员函数;
- const 成员函数中只能调用const成员函数;
- const 成员函数中不能直接改写成员变量的值。