欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++實(shí)現(xiàn)拷貝構(gòu)造函數(shù)的方法詳解

 更新時間:2022年09月20日 08:20:23   作者:Mi?ronin  
拷貝構(gòu)造函數(shù)是構(gòu)造函數(shù)的一個重載,因此顯式的定義了拷貝構(gòu)造,那么編譯器也不再默認(rèn)生成構(gòu)造函數(shù)。本文主要介紹了C++實(shí)現(xiàn)拷貝構(gòu)造函數(shù)的方法,需要的可以參考一下

引入

對于普通類型的對象來說,他們之間的復(fù)制很簡單:

int a = 10;
int b = a;

但是對于類對象來說,其中會存在許多的成員變量。

#include <iostream>
using namespace std;

class CExample {
private:
     int a;
public:
      //構(gòu)造函數(shù)
     CExample(int b)
     { a = b;}

      //一般函數(shù)
     void Show ()
     {
        cout<<a<<endl;
      }
};

int main()
{
     CExample A(100);
     CExample B = A; //注意這里的對象初始化要調(diào)用拷貝構(gòu)造函數(shù),而非賦值
      B.Show ();
     return 0;
}

從以上代碼可以看出系統(tǒng)為對象 B 分配了內(nèi)存并完成了與對象 A 的復(fù)制過程。就類對象而言,相同類型的類對象是通過拷貝構(gòu)造函數(shù)來完成整個復(fù)制過程的。

下面這個則是拷貝構(gòu)造函數(shù)的工作過程

#include <iostream>
using namespace std;

class CExample {
private:
    int a;
public:
    //構(gòu)造函數(shù)
    CExample(int b)
    { a = b;}
    
    //拷貝構(gòu)造函數(shù)
    CExample(const CExample& C)
    {
        a = C.a;
    }

    //一般函數(shù)
    void Show ()
    {
        cout<<a<<endl;
    }
};

int main()
{
    CExample A(100);
    CExample B = A; // CExample B(A); 也是一樣的
     B.Show ();
    return 0;
} 

在這里CExample(const CExample& C) 就是我們自定義的拷貝構(gòu)造函數(shù)。

一.什么是拷貝構(gòu)造函數(shù)

同一個類的對象在內(nèi)存中有完全相同的結(jié)構(gòu),如果作為一個整體進(jìn)行復(fù)制或稱拷貝是完全可行的。這個拷貝過程只需要拷貝數(shù)據(jù)成員,而函數(shù)成員是共用的(只有一份拷貝)。在建立對象時可用同一類的另一個對象來初始化該對象的存儲空間,這時所用的構(gòu)造函數(shù)稱為拷貝構(gòu)造函數(shù)。

拷貝構(gòu)造函數(shù)本質(zhì)上來說也是構(gòu)造函數(shù)。

二.什么情況下使用拷貝構(gòu)造函數(shù)

一般來說有以下三種情況:

  • 用舊對象去初始化新對象
  • 值傳遞—參數(shù)是類類型的值類型,從實(shí)參傳遞給形參的過程,是用實(shí)參去構(gòu)造形參
  • 函數(shù)返回值是值類型–用局部對象去構(gòu)造臨時對象調(diào)用拷貝構(gòu)造
class A
{
public:
    A(int i = 0):m_i(i)
    {
        cout<<"A(int) "<<m_i<<endl;
    }
    A(const A &a):m_i(a.m_i)
    {
        cout<<"A(A) "<<m_i<<endl;
    }
    ~A()
    {
        cout<<"~A "<<m_i<<endl;
    }
private:
    int m_i;
};
void fn(A t) //2.將c傳遞給t的過程,是值傳遞,此時臨時對象形參t是新對象,用c去構(gòu)造t,調(diào)用拷貝構(gòu)造
{
    cout<<"fn end"<<endl;
    //在退出fn函數(shù)時,將臨時對象t釋放,調(diào)用析構(gòu)函數(shù)
}
A test()
{
    A d(40); //調(diào)用普通構(gòu)造A(int)構(gòu)造對象d
    return d;
    /*
     3.在返回的時候,將局部對象的值給了臨時對象(值傳遞,調(diào)用拷貝構(gòu)造,用舊局部對象去構(gòu)造新的臨時對象)
     然后局部對象d就釋放了,臨時對象將值帶回到主調(diào)函數(shù)中后,臨時對象的值才釋放
    */
}
A fnn()
{
    A s(50); //A(int) 50
    return s; //
    /*1.s->臨時對象  拷貝構(gòu)造
    2.析構(gòu)函數(shù)局部對象s
    */
}
void main()
{
    A a(20);  //A(int) 20
    A b = a;  //1.用a初始化b A(A)
    cout<<"fn"<<endl;
    A c(30); //A(int) 30
    fn(c);
    cout<<"test"<<endl;
    c = test(); //臨時對象將值賦給c(調(diào)用賦值函數(shù))之后,調(diào)用析構(gòu),析構(gòu)臨時對象
    cout<<"fnn"<<endl;
    A t = fnn(); //臨時對象初始化新對象t,是否會調(diào)用拷貝構(gòu)造?--調(diào)用了,不過編譯器做過優(yōu)化
    cout<<"main end"<<endl;
}

三.使用拷貝構(gòu)造函數(shù)需要注意什么

拷貝構(gòu)造函數(shù)是構(gòu)造函數(shù)的一個重載形式。

拷貝構(gòu)造函數(shù)的參數(shù)只有一個且必須使用引用傳參,使用傳值方式會引發(fā)無窮遞歸調(diào)用。

若未顯示定義,系統(tǒng)生成默認(rèn)的拷貝構(gòu)造函數(shù)。默認(rèn)的拷貝構(gòu)造函數(shù)對象按內(nèi)存存儲按字節(jié)序完成拷貝,稱為:位拷貝。

四.深拷貝淺拷貝

4.1 淺拷貝

所謂淺拷貝,指的是在對象復(fù)制時,只對對象中的數(shù)據(jù)成員進(jìn)行簡單的賦值,默認(rèn)拷貝構(gòu)造函數(shù)執(zhí)行的也是淺拷貝。大多情況下“淺拷貝”已經(jīng)能很好地工作了,但是一旦對象存在了動態(tài)成員,指針,那么淺拷貝就會出現(xiàn)一些問題。

對于下面函數(shù)來說有指針作為數(shù)據(jù)成員,則用s1對象去構(gòu)造s2對象的時候,調(diào)用默認(rèn)拷貝構(gòu)造,用s1中的數(shù)據(jù)成員指針m_str去初始化s2對象中的數(shù)據(jù)成員m_str,即是s2.m_str = s1.m_str,那么導(dǎo)致兩個對象中的指針指向同一塊內(nèi)存單元,指向的都是構(gòu)造s1對象時開辟的內(nèi)存單元,所以在主函數(shù)退出時候要析構(gòu)s2和s1時,將同一段空間釋放兩次出現(xiàn)內(nèi)存錯誤。

class Str
{
public:
    Str(const char *str = "")
    {
        m_str = new char[strlen(str)+1];
        strcpy(m_str,str);
    }
    ~Str()
    {
        delete[]m_str;
    }
    void Print()
    {
        cout<<m_str<<endl;
    }
private:
    char *m_str;
};
void main()
{
    Str s1("pangpang");
    Str s2(s1);
    cout<<sizeof(Str)<<endl;
    s1.Print();
    s2.Print();
}

上述程序的內(nèi)存布局:

對于指針作為數(shù)據(jù)成員的類,用s1對象去構(gòu)造s2對象的時候,調(diào)用默認(rèn)拷貝構(gòu)造函數(shù)時,二者指向同一內(nèi)存單元,即二者的初始地址相同這里均為0X0002000,當(dāng)我們構(gòu)造完以后將要進(jìn)行析構(gòu)時,這里將會出現(xiàn)錯誤:因?yàn)槲鰳?gòu)函數(shù)要釋放空間,而這里我們的空間對應(yīng)的是一塊空間,當(dāng)我們析構(gòu)完s2后:這一塊空間的內(nèi)容已經(jīng)被delete,而我們還需要析構(gòu)s1,即:一個內(nèi)存空間析構(gòu)了兩次,出現(xiàn)內(nèi)存錯誤。

為了解決上述問題 我們就需要給s2中的m_str也開辟和s1中的m_str一樣大小的空間,所以我們就需要 深拷貝 。

4.2 深拷貝

在“深拷貝”的情況下,對于對象中動態(tài)成員,就不能僅僅簡單地賦值了,而應(yīng)該重新動態(tài)分配空間,如上面的例子就應(yīng)該按照如下的方式進(jìn)行處理:

class Str
{
public:
    Str(const char *str = "")
    {
        m_str = new char[strlen(str)+1];
        strcpy(m_str,str);
    }
    ~Str()
    {
        cout<<"~Str"<<endl;
        delete[]m_str;
    }
    Str(const Str& s)
    {
        cout<<"Str(Str)"<<endl;
        m_str = new char[strlen(s.m_str)+1]; //開辟同樣大小的空間
        strcpy(m_str,s.m_str); //將內(nèi)容拷貝成一樣的
    }
    void Print()
    {
        cout<<m_str<<endl;
    }
private:
    char *m_str;
};
void main()
{
    Str s1("pangpang");
    Str s2(s1); //拷貝構(gòu)造
    s1.Print();
    s2.Print();
}

現(xiàn)在的程序內(nèi)存布局為:

各自指向一段內(nèi)存空間,但它們指向的空間具有相同的內(nèi)容,這就是所謂的“深拷貝”。

到此這篇關(guān)于C++實(shí)現(xiàn)拷貝構(gòu)造函數(shù)的方法詳解的文章就介紹到這了,更多相關(guān)C++拷貝構(gòu)造函數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C語言完整特性詳情

    C語言完整特性詳情

    這篇文章主要介紹了C# 10的相關(guān)資料方法,感興趣的朋友可以參考下文
    2021-08-08
  • 基于C語言實(shí)現(xiàn)猜數(shù)字游戲

    基于C語言實(shí)現(xiàn)猜數(shù)字游戲

    這篇文章主要為大家詳細(xì)介紹了基于C語言實(shí)現(xiàn)猜數(shù)字游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-05-05
  • C++實(shí)現(xiàn)LeetCode(42.收集雨水)

    C++實(shí)現(xiàn)LeetCode(42.收集雨水)

    這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(42.收集雨水),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-07-07
  • C++動態(tài)內(nèi)存分配(new/new[]和delete/delete[])詳解

    C++動態(tài)內(nèi)存分配(new/new[]和delete/delete[])詳解

    這篇文章主要介紹了C++動態(tài)內(nèi)存分配(new/new[]和delete/delete[])詳解的相關(guān)資料,需要的朋友可以參考下
    2017-05-05
  • vscode刷acm、leetcode的題目

    vscode刷acm、leetcode的題目

    vscode是一款越來越受碼農(nóng)們喜愛的軟件,大多數(shù)人學(xué)習(xí)編程繞不開的一部分就是算法,很多人都喜歡刷LeetCode的題目,本文就來介紹一下
    2021-06-06
  • Qt實(shí)現(xiàn)文本編輯器(一)

    Qt實(shí)現(xiàn)文本編輯器(一)

    在Qt中QMainWindow是一個為用戶提供主窗口程序的類,包含了:菜單欄、工具欄、錨接部件、狀態(tài)欄以及一個中部件。本文將利用QMainWindow制作一個文本編輯器,感興趣的可以試一試
    2022-01-01
  • 在C++中反射調(diào)用.NET的方法(一)

    在C++中反射調(diào)用.NET的方法(一)

    為什么要在C++中調(diào)用.NET呢?接下來通過本文給大家介紹在C++中反射調(diào)用.NET的方法(一),需要的朋友參考下吧
    2017-02-02
  • C語言實(shí)現(xiàn)數(shù)字雨效果

    C語言實(shí)現(xiàn)數(shù)字雨效果

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)數(shù)字雨效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-02-02
  • C語言實(shí)現(xiàn)銀行管理系統(tǒng)(文件操作)

    C語言實(shí)現(xiàn)銀行管理系統(tǒng)(文件操作)

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)銀行管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • 如何用C語言、Python實(shí)現(xiàn)棧及典型應(yīng)用

    如何用C語言、Python實(shí)現(xiàn)棧及典型應(yīng)用

    本文先通過實(shí)例分別介紹了如何用C語言、Python實(shí)現(xiàn)棧,后又介紹棧的典型應(yīng)用,對大家學(xué)習(xí)棧很有借鑒參考價值,下面一起來看看吧。
    2016-08-08

最新評論