C++中构造函数、析构函数、拷贝构造函数详解

时间:2021-08-18 19:52:34

本篇文章主要讲解了C++中构造函数、析构函数、拷贝构造函数的容易产生理解上错误的地方。大家也可以作为检验对知识点掌握的情况!

首先代码如下:

class CStudent  
{
public:
  CStudent()
  {
  }
  CStudent(CStudent& stu)
  {
  }
  ~CStudent()
  {
  }
};

int main()
{
  CStudent stu1;

  CStudent stu2(stu1);
  // 这行代码如何解释?
  CStudent stu3();

  return 0; 
} 

 以上注释的那行代码该怎么解释?

实例化了一个对象并显示的调用了默认的构造函数?

如果真这么理解,那就大错特错了。其实这是一个函数声明,表示声明了一个返回值为CStudent、函数名为stu3的函数。

编译器在编译的过程中,因为是函数声明,并不产生实质性的代码!

继续来看代码。源码如下:

#include <iostream.h>


class CStudent  
{
public:
  CStudent()
  {
    m_iNum = 0;
    cout << "CStudent()" << endl;
  }
  CStudent(int iNum)
  {
    m_iNum = iNum;
    cout << "CStudent(int iNum)" << endl;
  }
  CStudent(CStudent& stu)
  {
    m_iNum = stu.m_iNum;
    cout << "CStudent(CStudent& stu)" << endl;
  }
  ~CStudent()
  {
    cout << "~CStudent()" << endl;
  }

  int m_iNum;
};

CStudent GetObject(CStudent obj)
{
  return obj.m_iNum;
}

int main()
{
  CStudent stu1;

  // 此处这行代码如何执行?
  GetObject(stu1);

  return 0; 
} 

首先,由于C++默认的是 __cdecl,也就是C调用约定。栈的平衡由调用方平衡

stu1是一个对象,GetObject的形参也是一个CStudent的对象,那么必然发生值的拷贝,换言之,会先调用obj对象的拷贝构造函数,然后在压入参数,保存返回地址。

在跳转到函数代码处执行,函数执行到return obj.m_iNum的时候,因为返回值是一个对象.所以。重载拷贝函数CStudent(int iNum)被调用,产生新(我们给它随便取个名字:“NewObj”对象,方便后面的讲解,继续往下执行,函数即将出作用域。此时,obj的析构函数被调用,然后程序返回出作用域。这时,继续往下执行,由于此行代码“;”为当前代码行作用域结束,NewObj对象的析构函数被调用!词条语句则执行完毕,如果在VC6.0中调试.以上这条语句执行后输出如下:

CStudent()

// 上面这行是实例化stu1对象的输

CStudent(CStudent& stu)
CStudent(int iNum)
~CStudent()
~CStudent()