显式调用构造函数产生的悲剧

简介:

昨天一个同学让我帮他调试程序,发现一个诡异的问题,明明一个类的私有成员被赋值了,但运行时却总是不定的值。仔细检查了好久,才发现原来他把构造函数当作普通函数进行了显式调用,结果产生了悲剧。

    

将他的问题简单的抽象如下:

    

大家看看下面这段代码的输出结果是什么?这段代码有问题么?


#include <iostream>  
 
class CTest  
{
  
public:
 
    CTest()  
    {  
        m_a = 1;  
    }  
    CTest(int b)  
    {  
        m_b = b;  
        CTest();  
    }  
    ~CTest()  
    {}  
      
    void show  
    {  
        std::cout << m_a << std::endl;  
        std::cout << m_b << std::endl;  
    }  
      
private:  
      
    int m_a;  
    int m_b;  
}; 
 
void main()  
{  
    CTest myTest(2);  
    myTest.show();  
}


-----------------------------------------------------------
【分析】
-----------------------------------------------------------


输出结果中,m_a是一个不确定的值,因为没有被赋初值,m_b 为2


注意下面这段代码


1
2
3
4
5
CTest( int  b)
{
    m_b = b;
    CTest();
}


在调用CTest()函数时,实际上是创建了一个匿名的临时CTest类对象,CTest()中赋值 m_a = 1 也是对该匿名对象赋值,故我们定义的myTest的m_a其实没有被赋值。说白了,其实构造函数并不像普通函数那样进行一段处理,而是创建了一个对象,并且对该对象赋初值,所以显式调用构造函数无法实现给私有成员赋值的目的。


这个例子告诉我们以后代码中千万不要出现使用一个构造函数显式调用另外一个构造函数,这样会出现不确定性。其实一些初始化的代码可以写在一个单独的init函数中,然后每一个构造函数都调用一下该初始化函数就行了。

在此,顺便再提出另外一个问题以供思考:


CTest *p = NULL;  
 
void func()  
 
{  
 
    p = new CTest();  
 
}

    

代码右边显示调用CTest(),是否依然会产生一个匿名的临时对象a,然后将该匿名的临时对象a的地址赋给指针p? 如果是这样的话,出了func函数后,临时对象a是否会被析构? 那指针p不成为了野指针了?你能解释这个问题么?



本文转自 Jhuster 51CTO博客,原文链接:http://blog.51cto.com/ticktick/294573,如需转载请自行联系原作者

相关文章
|
5月前
this的含义,什么情况下使用this,改变this指针的两种办法。 === 由于this关键字很混乱,如何解决这个问题
this的含义,什么情况下使用this,改变this指针的两种办法。 === 由于this关键字很混乱,如何解决这个问题
40 0
|
11月前
|
编译器 C语言 C++
一分钟搞懂什么是this指针(未涉及静态成员和函数)
一分钟搞懂什么是this指针(未涉及静态成员和函数)
|
编译器 C++
【一、构造函数与析构函数】深度解析C++类的构造函数与析构函数调用机制
【一、构造函数与析构函数】深度解析C++类的构造函数与析构函数调用机制
542 0
【错误记录】反射时调用方法及成员报错 ( 执行反射方法 | 设置反射的成员变量 | 设置方法/成员可见性 )
【错误记录】反射时调用方法及成员报错 ( 执行反射方法 | 设置反射的成员变量 | 设置方法/成员可见性 )
177 0
|
Kotlin
【Kotlin】Kotlin 构造函数 ( 主构造函数 | 主构造函数声明属性 | init 初始化代码块 | 次构造函数 | 构造函数委托 | 调用构造函数创建实例对象 )(一)
【Kotlin】Kotlin 构造函数 ( 主构造函数 | 主构造函数声明属性 | init 初始化代码块 | 次构造函数 | 构造函数委托 | 调用构造函数创建实例对象 )(一)
276 0
|
Kotlin
【Kotlin】Kotlin 构造函数 ( 主构造函数 | 主构造函数声明属性 | init 初始化代码块 | 次构造函数 | 构造函数委托 | 调用构造函数创建实例对象 )(二)
【Kotlin】Kotlin 构造函数 ( 主构造函数 | 主构造函数声明属性 | init 初始化代码块 | 次构造函数 | 构造函数委托 | 调用构造函数创建实例对象 )(二)
162 0
什么情况下会调用拷贝构造函数?
什么情况下会调用拷贝构造函数?
260 0