• C++构造函数详解及显式调用构造函数(explicit)


    一. 什么是拷贝构造函数

    首先对于普通类型的对象来说,它们之间的复制是很简单的,例如:

    int a = 100;  
    int b = a; 

    而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量。
    下面看一个类对象拷贝的简单例子。

    #include <iostream>  
    using namespace std;  
      
    class CExample {  
    private:  
         int a;  
    public:  
          //构造函数  
         CExample(int b)  
         { a = b;}  
      
          //一般函数  
         void Show ()  
         {  
            cout<<a<<endl;  
          }  
    };  
      
    int main()  
    {  
         CExample A(100);  
         CExample B = A; //注意这里的对象初始化要调用拷贝构造函数,而非赋值  
          B.Show ();  
         return 0;  
    }  

    运行程序,屏幕输出100。从以上代码的运行结果可以看出,系统为对象 B 分配了内存并完成了与对象 A 的复制过程。就类对象而言,相同类型的类对象是通过拷贝构造函数来完成整个复制过程的。

    下面举例说明拷贝构造函数的工作过程。

    #include <iostream>  
    using namespace std;  
      
    class CExample {  
    private:  
        int a;  
    public:  
        //构造函数  
        CExample(int b)  
        { a = b;}  
          
        //拷贝构造函数  
        CExample(const CExample& C)  
        {  
            a = C.a;  
        }  
      
        //一般函数  
        void Show ()  
        {  
            cout<<a<<endl;  
        }  
    };  
      
    int main()  
    {  
        CExample A(100);  
        CExample B = A; // CExample B(A); 也是一样的  
         B.Show ();  
        return 0;  
    }   


    CExample(const CExample& C) 就是我们自定义的拷贝构造函数。可见,拷贝构造函数是一种特殊的构造函数,函数的名称必须和类名称一致,它必须的一个参数是本类型的一个引用变量

    二. 拷贝构造函数的调用时机

    在C++中,下面三种对象需要调用拷贝构造函数!
    1. 对象以值传递的方式传入函数参数

    class CExample   
    {  
    private:  
     int a;  
      
    public:  
     //构造函数  
     CExample(int b)  
     {   
      a = b;  
      cout<<"creat: "<<a<<endl;  
     }  
      
     //拷贝构造  
     CExample(const CExample& C)  
     {  
      a = C.a;  
      cout<<"copy"<<endl;  
     }  
       
     //析构函数  
     ~CExample()  
     {  
      cout<< "delete: "<<a<<endl;  
     }  
      
         void Show ()  
     {  
             cout<<a<<endl;  
         }  
    };  
      
    //全局函数,传入的是对象  
    void g_Fun(CExample C)  
    {  
     cout<<"test"<<endl;  
    }  
      
    int main()  
    {  
     CExample test(1);  
     //传入对象  
     g_Fun(test);  
      
     return 0;  
    }  


    调用g_Fun()时,会产生以下几个重要步骤:
    (1).test对象传入形参时,会先会产生一个临时变量,就叫 C 吧。
    (2).然后调用拷贝构造函数把test的值给C。 整个这两个步骤有点像:CExample C(test);
    (3).等g_Fun()执行完后, 析构掉 C 对象。

    2. 对象以值传递的方式从函数返回

    class CExample   
    {  
    private:  
     int a;  
      
    public:  
     //构造函数  
     CExample(int b)  
     {   
      a = b;  
     }  
      
     //拷贝构造  
     CExample(const CExample& C)  
     {  
      a = C.a;  
      cout<<"copy"<<endl;  
     }  
      
         void Show ()  
         {  
             cout<<a<<endl;  
         }  
    };  
      
    //全局函数  
    CExample g_Fun()  
    {  
     CExample temp(0);  
     return temp;  
    }  
      
    int main()  
    {  
     g_Fun();  
     return 0;  
    }  


    当g_Fun()函数执行到return时,会产生以下几个重要步骤:
    (1). 先会产生一个临时变量,就叫XXXX吧。
    (2). 然后调用拷贝构造函数把temp的值给XXXX。整个这两个步骤有点像:CExample XXXX(temp);
    (3). 在函数执行到最后先析构temp局部变量。
    (4). 等g_Fun()执行完后再析构掉XXXX对象。

    3. 对象需要通过另外一个对象进行初始化;

    CExample A(100);  
    CExample B = A;   
    // CExample B(A);   

    后两句都会调用拷贝构造函数。

    以上原文地址:https://blog.csdn.net/lwbeyond/article/details/6202256

    补充:

    #include<iostream>
    using namespace std;
    
    class A
    {
    public:
    	A(const A& )
    	{
    		cout << "copy"<<endl;
    	}
    	A()
    	{
    		i++;
    	}
    	virtual ~A()
    	{
    		i--;
    	}
    	void fun()
    	{
    		cout << i<< " ";	
    	}
    	static int i;
    };
    
    int A::i = 0;
    
    void fun(A a) {  // 如果是void fun(A &a)即引用,将不会调用拷贝构造函数,调用结束也就没有析构了
    	static A aa;  // A::i变成3,初始化1次,且整个程序运行结束时析构,如无static则退出fun函数就会析构
    	a.fun();
    }
    int main()
    {	
    	A *p = new A; // A::i变成1
    	A a;         // A::i变成2
    	fun(a);  // 会调用拷贝构造函数,调用之后会调用一次析构
    	fun(a);  // 会调用拷贝构造函数,调用之后会调用一次析构
    	delete p;  // 析构1次
    	fun(a);   // 会调用拷贝构造函数,调用之后会调用一次析构
    	
    	return 0;
    }

    输出:

    一、 构造函数是干什么的
    class Counter
    {
    public:
             // 类Counter的构造函数
             // 特点:以类名作为函数名,无返回类型
             Counter()
             {
                    m_value = 0;
             }
    private:
              // 数据成员
             int m_value;
    }
           该类对象被创建时,编译系统对象分配内存空间,并自动调用该构造函数->由构造函数完成成员的初始化工作
    eg:    Counter c1;
           编译系统为对象c1的每个数据成员(m_value)分配内存空间,并调用构造函数Counter( )自动地初始化对象c1的m_value值设置为0
    故:
            构造函数的作用:初始化对象的数据成员。
    二、 构造函数的种类
    class Complex 
    {         
    private :
            double    m_real;
            double    m_imag;
    public:
            //    无参数构造函数
            // 如果创建一个类你没有写任何构造函数,则系统会自动生成默认的无参构造函数,函数为空,什么都不做
            // 只要你写了一个下面的某一种构造函数,系统就不会再自动生成这样一个默认的构造函数,如果希望有一个这样的无参构造函数,则需要自己显示地写出来
            Complex(void)
            {
                 m_real = 0.0;
                 m_imag = 0.0;
            }   
            //    一般构造函数(也称重载构造函数)
            // 一般构造函数可以有各种参数形式,一个类可以有多个一般构造函数,前提是参数的个数或者类型不同(基于c++的重载函数原理)
            // 例如:你还可以写一个 Complex( int num)的构造函数出来
            // 创建对象时根据传入的参数不同调用不同的构造函数
            Complex(double real, double imag)
            {
                 m_real = real;
                 m_imag = imag;         
             }  
            //    复制构造函数(也称为拷贝构造函数)
            //    复制构造函数参数为类对象本身的引用,用于根据一个已存在的对象复制出一个新的该类的对象,一般在函数中会将已存在对象的数据成员的值复制一份到新创建的对象中
            //    若没有显示的写复制构造函数,则系统会默认创建一个复制构造函数,但当类中有指针成员时,由系统默认创建该复制构造函数会存在风险,具体原因请查询 有关 “浅拷贝” 、“深拷贝”的文章论述
            Complex(const Complex & c)
            {
                    // 将对象c中的数据成员值复制过来
                    m_real = c.m_real;
                    m_imag    = c.m_imag;
            }            
            // 类型转换构造函数,根据一个指定的类型的对象创建一个本类的对象
         //需要注意的一点是,这个其实就是一般的构造函数,但是对于出现这种单参数的构造函数,C++会默认将参数对应的类型转换为该类类型,有时候这种隐私的转换是我们所不想要的,所以需要使用explicit来限制这种转换。

           // 例如:下面将根据一个double类型的对象创建了一个Complex对象

           Complex(double r)
            {
                    m_real = r;
                    m_imag = 0.0;
            }
            // 等号运算符重载也叫赋值构造函数)
            // 注意,这个类似复制构造函数,将=右边的本类对象的值复制给等号左边的对象,它不属于构造函数,等号左右两边的对象必须已经被创建
            // 若没有显示的写=运算符重载,则系统也会创建一个默认的=运算符重载,只做一些基本的拷贝工作
            Complex &operator=( const Complex &rhs )
            {
                    // 首先检测等号右边的是否就是左边的对象本身,若是本对象本身,则直接返回
                    if ( this == &rhs ) 
                    {
                            return *this;
                    }                
                    // 复制等号右边的成员到左边的对象中
                    this->m_real = rhs.m_real;
                    this->m_imag = rhs.m_imag;                
                   // 把等号左边的对象再次传出
                   // 目的是为了支持连等 eg:    a=b=c 系统首先运行 b=c
                   // 然后运行 a= ( b=c的返回值,这里应该是复制c值后的b对象)    
                    return *this;
            }
    };

    下面使用上面定义的类对象来说明各个构造函数的用法:
    int main()
    {
            // 调用了无参构造函数,数据成员初值被赋为0.0
            Complex c1,c2;

            // 调用一般构造函数,数据成员初值被赋为指定值
            Complex c3(1.0,2.5);
            // 也可以使用下面的形式
            Complex c3 = Complex(1.0,2.5);
            
            //    把c3的数据成员的值赋值给c1
            //    由于c1已经事先被创建,故此处不会调用任何构造函数
            //    只会调用 = 号运算符重载函数
            c1 = c3;        
            //    调用类型转换构造函数
            //    系统首先调用类型转换构造函数,将5.2创建为一个本类的临时对象,然后调用等号运算符重载,将该临时对象赋值给c1
            c2 = 5.2;        
           // 调用拷贝构造函数( 有下面两种调用方式) 
            Complex c5(c2);
            Complex c4 = c2;  // 注意和 = 运算符重载区分,这里等号左边的对象不是事先已经创建,故需要调用拷贝构造函数,参数为c2
    //这一点特别重要,这儿是初始化,不是赋值。其实这儿就涉及了C++中的两种初始化的方式:复制初始化和赋值初始化。其中c5采用的是复制初始化,而c4采用的是赋值初始化,这两种方式都是要调用拷贝构造函数的。
    }

    三、思考与测验
    1. 仔细观察复制构造函数
            Complex(const Complex & c)
            {
                    // 将对象c中的数据成员值复制过来
                    m_real = c.m_real;
                    m_img = c.m_img;
            }        
    为什么函数中可以直接访问对象c的私有成员?
    答:(网上)因为拷贝构造函数是放在本身这个类里的,而类中的函数可以访问这个类的对象的所有成员,当然包括私有成员了。
    2. 挑战题,了解引用与传值的区别
      Complex test1(const Complex& c)
      {
              return c;
      }
       Complex test2(const Complex c)
      {
             return c;
       }
        Complex test3()
       {
              static Complex c(1.0,5.0);
              return c;
       }
       Complex& test4()
      {
             static Complex c(1.0,5.0);
             return c;
      }  
      void main()
      {
            Complex a,b;    
            // 下面函数执行过程中各会调用几次构造函数,调用的是什么构造函数?    
           test1(a);
           test2(a);     
           b = test3();
           b = test4();     
           test2(1.2);
           // 下面这条语句会出错吗?
           test1(1.2);     //test1( Complex(1.2 )) 呢?
      }
      答:

    为了便于看构造函数的调用效果,我将类重新改一下,添加一些输出信息,代码如下:

    #include <iostream>
    using namespace std;
    class Complex
    {        
        private :
        double    m_real;
        double    m_imag;
        int id;
        static int counter;
        public:
        //    无参数构造函数
        Complex(void)
        {
            m_real = 0.0;
            m_imag = 0.0;
            id=(++counter);
            cout<<"Complex(void):id="<<id<<endl;
        }
        //    一般构造函数(也称重载构造函数)
        Complex(double real, double imag)
        {
            m_real = real;
            m_imag = imag;        
            id=(++counter);
            cout<<"Complex(double,double):id="<<id<<endl;
        }
        //    复制构造函数(也称为拷贝构造函数)
        Complex(const Complex & c)
        {
            // 将对象c中的数据成员值复制过来
            m_real = c.m_real;
            m_imag = c.m_imag;
            id=(++counter);
            cout<<"Complex(const Complex&):id="<<id<<" from id="<<c.id<<endl;
        }            
        // 类型转换构造函数,根据一个指定的类型的对象创建一个本类的对象
        Complex(double r)
        {
            m_real = r;
            m_imag = 0.0;
            id=(++counter);
            cout<<"Complex(double):id="<<id<<endl;
        }
        ~Complex()
        {
            cout<<"~Complex():id="<<id<<endl;
        }
        // 等号运算符重载
        Complex &operator=( const Complex &rhs )
        {
            if ( this == &rhs ) {
                return *this;
            }
            this->m_real = rhs.m_real;
            this->m_imag = rhs.m_imag;
            cout<<"operator=(const Complex&):id="<<id<<" from id="<<rhs.id<<endl;
            return *this;
        }
    };
    int Complex::counter=0;
    Complex test1(const Complex& c)
    {
        return c;
    }
    Complex test2(const Complex c)
    {
        return c;
    }
    Complex test3()
    {
        static Complex c(1.0,5.0);
        return c;
    }
    Complex& test4()
    {
        static Complex c(1.0,5.0);
        return c;
    }
    int main()
    {
        Complex a,b;
    
        // 下面函数执行过程中各会调用几次构造函数,调用的是什么构造函数?
        Complex c=test1(a);
        Complex d=test2(a);
    
        b = test3();
        b = test4();
    
        Complex e=test2(1.2);
        Complex f=test1(1.2);
        Complex g=test1(Complex(1.2));
        return 0;
    }

    下面是程序运行结果:第一次运行结果:

    第二次运行结果:

    第三次运行结果:

    四、附录(浅拷贝与深拷贝)

        上面提到,如果没有自定义复制构造函数,则系统会创建默认的复制构造函数,但系统创建的默认复制构造函数只会执行“浅拷贝”,即将被拷贝对象的
    数据成员的 值一一赋值给新创建的对象,若该类的数据成员中有指针成员,则会使得新的对象的指针所指向的地址与被拷贝对象的指针所指向的地址相同,
    delete该指针 时则会导致两次重复delete而出错。下面是示例:   

     【浅拷贝与深拷贝】

    #include <iostream.h>
    #include <string.h>
    class Person 
    {
    public :
             // 构造函数
            Person(char * pN)
            {
                  cout << "一般构造函数被调用 ! ";
                  m_pName = new char[strlen(pN) + 1];
                  //在堆中开辟一个内存块存放pN所指的字符串
                  if(m_pName != NULL) 
                  {
                     //如果m_pName不是空指针,则把形参指针pN所指的字符串复制给它
                       strcpy(m_pName ,pN);
                  }
            }        
            
            // 系统创建的默认复制构造函数,只做位模式拷贝
            Person(Person & p)    
            { 
                      //使两个字符串指针指向同一地址位置         
                     m_pName = p.m_pName;         
            }
             ~Person( )
            {
                    delete m_pName;
            }
      private :

            char * m_pName;
    };

    void main( )

            Person man("lujun");
            Person woman(man); 
            
            // 结果导致   man 和    woman 的指针都指向了同一个地址
            
            // 函数结束析构时
            // 同一个地址被delete两次
    }
    // 下面自己设计复制构造函数,实现“深拷贝”,即不让指针指向同一地址,而是重新申请一块内存给新的对象的指针数据成员
    Person(Person & chs);
    {
             // 用运算符new为新对象的指针数据成员分配空间
             m_pName=new char[strlen(p.m_pName)+ 1];

             if(m_pName)         
             {
                     // 复制内容
                    strcpy(m_pName ,chs.m_pName);
             }
          
            // 则新创建的对象的m_pName与原对象chs的m_pName不再指向同一地址了
    }

    参考地址:http://ticktick.blog.51cto.com/823160/194307

    下面讨论一个重要问题是:构造函数的显式调用

     

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

     #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

    注意下面这段代码
    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不成为了野指针了?你能解释这个问题么?

    答:我实验的结果是不会产生临时对象a,直接将产生的对象指针赋给了p

    new参考:https://blog.csdn.net/a3192048/article/details/80213288

    参考:http://ticktick.blog.51cto.com/823160/294573

     

    原文地址:https://www.cnblogs.com/xkfz007/archive/2012/05/11/2496447.html

  • 相关阅读:
    Mac 10.12安装Atom文本增强编辑工具
    Mac 10.12安装SecureCRT
    Mac 10.12安装WebStorm
    Mac 10.12安装Command+Q误按提示工具
    Mac 10.12安装FTP工具FileZilla
    Mac 10.12安装VirtualBox
    Mac 10.12安装数据库管理工具MySQL Workbench
    Mac 10.12安装Homebrew图形化界面管理工具Cakebrew
    Mac 10.12安装图片切换工具ArcSoft Photo+
    Mac 10.12安装Git管理工具SourceTree
  • 原文地址:https://www.cnblogs.com/a3192048/p/12241335.html
Copyright © 2020-2023  润新知