C++拷貝函數的實現及用法

一、拷貝函數概述

拷貝函數是C++中的類成員函數之一,用於在對象創建時或對象作為函數參數傳遞時,將一個對象的值複製給另一個對象。當沒有定義拷貝函數時,C++編譯器將自動生成一個淺複製的默認拷貝函數。默認拷貝函數的作用是將一個對象的成員變量值複製給另一個對象,但對於指針類型等複雜類型,這種複製方式顯然是不合適的。

二、淺拷貝和深拷貝的區別

淺拷貝是指在執行拷貝過程時,只是將內存中的數據進行簡單的複製,這樣將得到兩個指針指向同一塊內存地址。

深拷貝是指在執行拷貝過程時,將自定義類型的內存塊複製到新的內存塊中,從而這兩個對象互不干擾。

//示例代碼
#include 
#include 

using namespace std;

class String {
public:
    String(const char* str = NULL);//普通構造函數 
    String(const String &other);//拷貝構造函數
    ~String();//析構函數
    String& operator =(const String& other);//賦值函數
private:
    char *m_data;//用於保存字符串
    int m_len;//用於保存字符串長度
};

String::String(const char* str) {
    if (str == NULL) {
        m_data = new char[1];
        *m_data = '\0';
        m_len = 0;
    }
    else {
        m_len = strlen(str);
        m_data = new char[m_len + 1];
        strcpy(m_data, str);
    }
}

String::String(const String &other) {
    m_len = other.m_len;
    m_data = new char[m_len + 1];
    strcpy(m_data, other.m_data);
}

String::~String() {
    delete[] m_data;
}

String& String::operator =(const String& other) {
    if (this == &other)//檢查自賦值
        return *this;
    delete[] m_data;//釋放舊空間
    m_len = other.m_len;
    m_data = new char[m_len + 1];
    strcpy(m_data, other.m_data);
    return *this;
}

三、深拷貝的實現

為了實現深拷貝,我們需要自定義拷貝函數(拷貝構造函數、賦值函數),以避免默認拷貝函數的淺複製行為。拷貝構造函數用於在對象創建時複製一個現有對象,賦值函數則用於在對象已經存在時將一個現有對象的值複製給另一個對象。下面的代碼就是一個使用深拷貝的例子。

//示例代碼
#include 
#include 

using namespace std;

class Student {
public:
    Student();
    Student(const char *name, int age, const char *gender);
    Student(const Student &stu);
    ~Student();

    const char* getName() const;
    int getAge() const;
    const char* getGender() const;

    void setName(const char *name);
    void setAge(int age);
    void setGender(const char *gender);

    Student& operator=(const Student &stu);

private:
    char *m_name;
    int m_age;
    char *m_gender;
};

Student::Student(): m_name(NULL), m_age(0), m_gender(NULL) {}

Student::Student(const char *name, int age, const char *gender) {
    if(name) {
        m_name = new char[strlen(name) + 1];
        strcpy(m_name, name);
    } else {
        m_name = new char[1];
        *m_name = '\0';
    }

    m_age = age;

    if(gender) {
        m_gender = new char[strlen(gender) + 1];
        strcpy(m_gender, gender);
    } else {
        m_gender = new char[1];
        *m_gender = '\0';
    }
}

Student::Student(const Student &stu) {
    if(stu.m_name) {
        m_name = new char[strlen(stu.m_name) + 1];
        strcpy(m_name, stu.m_name);
    } else {
        m_name = new char[1];
        *m_name = '\0';
    }

    m_age = stu.m_age;

    if(stu.m_gender) {
        m_gender = new char[strlen(stu.m_gender) + 1];
        strcpy(m_gender, stu.m_gender);
    } else {
        m_gender = new char[1];
        *m_gender = '\0';
    }
}

Student::~Student() {
    if(m_name) {
        delete[] m_name;
    }
    if(m_gender) {
        delete[] m_gender;
    }
}

const char* Student::getName() const {
    return m_name;
}

int Student::getAge() const {
    return m_age;
}

const char* Student::getGender() const {
    return m_gender;
}

void Student::setName(const char *name) {
    if(name) {
        m_name = new char[strlen(name) + 1];
        strcpy(m_name, name);
    } else {
        m_name = new char[1];
        *m_name = '\0';
    }
}

void Student::setAge(int age) {
    m_age = age;
}

void Student::setGender(const char *gender) {
    if(gender) {
        m_gender = new char[strlen(gender) + 1];
        strcpy(m_gender, gender);
    } else {
        m_gender = new char[1];
        *m_gender = '\0';
    }
}

Student& Student::operator=(const Student &stu) {
    if(this == &stu) {// 檢查自賦值
        return *this;
    }

    if(m_name) {
        delete[] m_name;
    }

    if(stu.m_name) {
        m_name = new char[strlen(stu.m_name) + 1];
        strcpy(m_name, stu.m_name);
    } else {
        m_name = new char[1];
        *m_name = '\0';
    }

    m_age = stu.m_age;

    if(m_gender) {
        delete[] m_gender;
    }

    if(stu.m_gender) {
        m_gender = new char[strlen(stu.m_gender) + 1];
        strcpy(m_gender, stu.m_gender);
    } else {
        m_gender = new char[1];
        *m_gender = '\0';
    }

    return *this;
}

四、拷貝函數的使用場景

在以下場景中,拷貝函數尤為重要:

1、當類具有指針成員時,必須使用深拷貝,以避免淺複製時指針成員指向同一個地址。

2、當類中的成員總量較大時,賦值操作會佔據很多時間,因此深拷貝函數可以顯著提高代碼的執行效率。

3、當類中包含了單例模式時,拷貝構造函數和賦值構造函數就有着更為重要的意義。

五、總結

在C++編程中,拷貝函數是相當重要的一個內容。在類的實例化以及類實例的拷貝過程中,深拷貝有時還需要重寫拷貝函數來保證正確性。因此,C++工程師必須充分理解拷貝函數的內涵,能夠正確地實現它,以避免因深複製的不當實現而導致程序的不穩定。

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-hk/n/304552.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2025-01-01 11:05
下一篇 2025-01-01 11:05

相關推薦

  • Python中引入上一級目錄中函數

    Python中經常需要調用其他文件夾中的模塊或函數,其中一個常見的操作是引入上一級目錄中的函數。在此,我們將從多個角度詳細解釋如何在Python中引入上一級目錄的函數。 一、加入環…

    編程 2025-04-29
  • Python中capitalize函數的使用

    在Python的字符串操作中,capitalize函數常常被用到,這個函數可以使字符串中的第一個單詞首字母大寫,其餘字母小寫。在本文中,我們將從以下幾個方面對capitalize函…

    編程 2025-04-29
  • Python中set函數的作用

    Python中set函數是一個有用的數據類型,可以被用於許多編程場景中。在這篇文章中,我們將學習Python中set函數的多個方面,從而深入了解這個函數在Python中的用途。 一…

    編程 2025-04-29
  • 單片機打印函數

    單片機打印是指通過串口或並口將一些數據打印到終端設備上。在單片機應用中,打印非常重要。正確的打印數據可以讓我們知道單片機運行的狀態,方便我們進行調試;錯誤的打印數據可以幫助我們快速…

    編程 2025-04-29
  • 三角函數用英語怎麼說

    三角函數,即三角比函數,是指在一個銳角三角形中某一角的對邊、鄰邊之比。在數學中,三角函數包括正弦、餘弦、正切等,它們在數學、物理、工程和計算機等領域都得到了廣泛的應用。 一、正弦函…

    編程 2025-04-29
  • Python3定義函數參數類型

    Python是一門動態類型語言,不需要在定義變量時顯示的指定變量類型,但是Python3中提供了函數參數類型的聲明功能,在函數定義時明確定義參數類型。在函數的形參後面加上冒號(:)…

    編程 2025-04-29
  • Python實現計算階乘的函數

    本文將介紹如何使用Python定義函數fact(n),計算n的階乘。 一、什麼是階乘 階乘指從1乘到指定數之間所有整數的乘積。如:5! = 5 * 4 * 3 * 2 * 1 = …

    編程 2025-04-29
  • Python定義函數判斷奇偶數

    本文將從多個方面詳細闡述Python定義函數判斷奇偶數的方法,並提供完整的代碼示例。 一、初步了解Python函數 在介紹Python如何定義函數判斷奇偶數之前,我們先來了解一下P…

    編程 2025-04-29
  • Python函數名稱相同參數不同:多態

    Python是一門面向對象的編程語言,它強烈支持多態性 一、什麼是多態多態是面向對象三大特性中的一種,它指的是:相同的函數名稱可以有不同的實現方式。也就是說,不同的對象調用同名方法…

    編程 2025-04-29
  • 分段函數Python

    本文將從以下幾個方面詳細闡述Python中的分段函數,包括函數基本定義、調用示例、圖像繪製、函數優化和應用實例。 一、函數基本定義 分段函數又稱為條件函數,指一條直線段或曲線段,由…

    編程 2025-04-29

發表回復

登錄後才能評論