有趣的问题:

 

我们在无参构造函数中调用带一个参数的构造函数。

程序如下:

 1 #include <stdio.h>
 2 
 3 class Test {
 4     int mi;
 5 public:
 6     Test(int i) {
 7         mi = i;
 8     }
 9     Test() {
10         Test(0);
11     }
12     void print() {
13         printf("mi = %d\n", mi);
14     }
15 };
16 
17 
18 int main()
19 {
20     Test t;
21     
22     t.print();
23 
24     return 0;
25 }

运行结果如下:

 

mi并没有打印出0,而是一个随机值。

发生了什么呢?

 

思考:

构造函数可以直接手工调用。

上面的程序第10行直接调用了构造函数产生了临时对象。它的生命期只有第10行这一条语句,过了这条语句就会被析构。

而且这个临时对象没有名字,作用域也仅仅在第10行这条语句中。第10行这条语句等价于没有,也就是空语句,因为这个临时对象我们无法使用。

我们在无参构造函数中调用有参构造函数的本意是代码复用,但是没有达到目的,在工程中,代码复用时,我们要在类中提供一个普通的init函数,这时就可以将公共代码写到这个函数中。

 临时对象示例:

 

由临时对象直接调用成员函数:

 

可以看到临时对象构造完之后立即调用了打印函数。

工程中要避免临时对象的产生与使用。

编译器的行为:

现代编译器在不影响最终执行结果的前提下,会尽力减少临时对象的产生。

示例如下:

 1 #include <stdio.h>
 2 
 3 class Test
 4 {
 5     int mi;
 6 public:
 7     Test(int i)
 8     {
 9         printf("Test(int i) : %d\n", i);
10         mi = i;
11     }
12     Test(const Test& t)
13     {
14         printf("Test(const Test& t) : %d\n", t.mi);
15         mi = t.mi;
16     }
17     Test()
18     {
19         printf("Test()\n");
20         mi = 0;
21     }
22     int print()
23     {
24         printf("mi = %d\n", mi);
25     }
26     ~Test()
27     {
28         printf("~Test()\n");
29     }
30 };
31 
32 Test func()
33 {
34     return Test(20);
35 }
36 
37 int main()
38 {
39     Test t = Test(10); // ==> Test t = 10;
40     Test tt = func();  // ==> Test tt = Test(20); ==> Test tt = 20;
41     
42     t.print();
43     tt.print();
44     
45     return 0;
46 }

39行的Test t = 10和Test t = Test(10)的编译运行结果是一样的。

Test t = Test(10)在我们的直觉判断会有以下两步,1、创建临时对象;2、用临时对象初始化t(调用拷贝构造函数)。

但是程序执行结果并不是按照我们的直觉分析那样的,这就是因为编译器做了优化。编译器会尽力减少临时对象的产生。

编译器将Test t = Test(10)等价成了Test t = 10的形式。

临时对象的产生会带来性能上面问题。

工程中的构造函数会非常复杂,调用一次构造函数耗时很长。

上述程序的运行结果如下:

 

 

32-35行返回临时对象,然后在40行用这个对象初始化tt,但是从打印结果可以看出,构造函数并不是按照我们想象的情形来调用的,我们想象的情形是先调用构造函数产生临时对象,然后调用拷贝构造函数初始化tt。运行结果没有调用拷贝构造函数。

这也是因为编译器的优化,在不影响执行结果的前提下会极力的减少临时对象的产生。

我们写程序也应该尽力的避开临时对象,例如程序写成Test t = 10,而不是Test t  = Test(10)。这样就可以保证在任何一款编译器下都不会产生临时对象。

小结:

 

posted on 2018-09-01 11:55  周伯通789  阅读(120)  评论(0编辑  收藏  举报