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

C++類中的運(yùn)算符重載過(guò)程

 更新時(shí)間:2024年11月13日 11:32:13   作者:gemluoye  
文章介紹了運(yùn)算符重載在C++中的重要性以及實(shí)現(xiàn)方法,包括加法運(yùn)算符重載、左移運(yùn)算符重載、遞增運(yùn)算符重載、+=運(yùn)算符重載、關(guān)系運(yùn)算符重載和賦值運(yùn)算符重載

為什么要實(shí)現(xiàn)運(yùn)算符重載?

在 C++ 中,運(yùn)算符最初是為內(nèi)置類型(如int、double等)定義操作方式。當(dāng)定義一個(gè)新的類(自定義類型)時(shí),編譯器并不清楚如何對(duì)這個(gè)新類型的成員變量應(yīng)用運(yùn)算符。

通過(guò)運(yùn)算符重載,程序員可以明確地告訴編譯器對(duì)于該類的對(duì)象,運(yùn)算符(如+、-、*等)應(yīng)該如何操作其成員變量,從而使自定義類型能夠像內(nèi)置類型一樣自然地使用這些運(yùn)算符。

1.加法運(yùn)算符重載

作用:實(shí)現(xiàn)兩個(gè)自定義類型數(shù)據(jù)的加法運(yùn)算

在類內(nèi)實(shí)現(xiàn)加法運(yùn)算符重載

#include <iostream>
using namespace std;
class Person {
    int age;
    int money;

public:
    Person() : money(0), age(0) {}
    Person(int val, int val2) : age(val), money(val2) {}
    Person(const Person& other) {
        this->age = other.age;
        this->money = other.money;
    }
    // 類內(nèi)實(shí)現(xiàn)+重載 本質(zhì):Person p3=p1.operator+(p2)
    Person operator+(const Person& other) {
        Person p;
        p.age = this->age + other.age;
        p.money = this->money + other.money;
        return p; // 不以引用返回是因?yàn)?,在?zhí)行完這個(gè)函數(shù)后,p會(huì)被銷毀,以值傳遞會(huì)調(diào)用拷貝構(gòu)造
    }
    void print() { cout << age << " " << money << endl; }
};

int main() {
    Person a(18, 50);    // 括號(hào)法
    Person b = {12, 50}; // 隱式轉(zhuǎn)換法
    Person c = a + b;
    c.print();
    return 0;
}

實(shí)現(xiàn)加法運(yùn)算符重載的operator+函數(shù)來(lái)完成的,傳入的參數(shù)為const Person& other的原因有以下的兩點(diǎn):

  • 1.以引用的方式傳遞是為了防止調(diào)用拷貝構(gòu)造
  • 2.加const修飾是為了防止修改實(shí)參

在函數(shù)中聲明了一個(gè)Person的對(duì)象p,返回值是以值的形式返回,因?yàn)橹捣祷貢?huì)調(diào)用拷貝構(gòu)造,如果是以引用的方式傳遞,當(dāng)這個(gè)函數(shù)結(jié)束時(shí),對(duì)象p就會(huì)被銷毀掉。

在類外實(shí)現(xiàn)加法運(yùn)算符重載

#include <iostream>
using namespace std;
class Person {
    int age;
    int money;
    friend Person operator+(const Person& other1, const Person& other2);

public:
    Person() : money(0), age(0) {}
    Person(int val, int val2) : age(val), money(val2) {}
    Person(const Person& other) {
        this->age = other.age;
        this->money = other.money;
    }
    void print() { cout << age << " " << money << endl; }
};
// 類外實(shí)現(xiàn)+重載 本質(zhì):Person p3=operator+(p1,p2)
Person operator+(const Person& other1, const Person& other2) {
    Person p;
    p.age = other1.age + other2.age;
    p.money = other1.money + other2.money;
    return p;
}
int main() {
    Person a(18, 50);    // 括號(hào)法
    Person b = {12, 50}; // 隱式轉(zhuǎn)換法
    Person c = a + b;
    c.print();
    return 0;
}

在類外實(shí)現(xiàn)加法運(yùn)算符重載,相當(dāng)于類外函數(shù)訪問類內(nèi)的成員變量,所以要將這個(gè)函數(shù)在類中聲明為友元函數(shù)(friend)。其他的跟在類內(nèi)實(shí)現(xiàn)加法運(yùn)算符重載是一樣的。

本質(zhì):在類內(nèi)實(shí)現(xiàn)加法運(yùn)算符重載的本質(zhì)是:Person p3=p1.operator(p2)

在類外實(shí)現(xiàn)加法運(yùn)算符重載的本質(zhì)是:Person p3=operator+(p1,p2)

因?yàn)橐粋€(gè)是通過(guò)對(duì)象去調(diào)用這個(gè)函數(shù),它本身也算是一個(gè)參數(shù),所以只需要傳人一個(gè)參數(shù)。而在類外,則是相當(dāng)于直接調(diào)用這個(gè)函數(shù),所以傳入的參數(shù)是兩個(gè)。

無(wú)論是再類內(nèi)還是在類外實(shí)現(xiàn)加法運(yùn)算符重載,最后調(diào)用的方式都是+。

2.左移運(yùn)算符重載

作用:可以輸出自定義數(shù)據(jù)類型

要在類外實(shí)現(xiàn),因?yàn)樵陬悆?nèi)的話,調(diào)用格式為:對(duì)象.operator();無(wú)法達(dá)到cout<<

#include <iostream>
using namespace std;
class Person {
    int age;
    int money;

public:
    friend ostream& operator<<(ostream& o, const Person& p);
    Person() : age(0), money(0) {}
    Person(int val, int val2) : age(val), money(val2) {}
    Person(const Person& other) {
        cout << "調(diào)用拷貝構(gòu)造";
        this->age = other.age;
        this->money = other.money;
    }
    // 一般不在函數(shù)內(nèi)進(jìn)行左移重載
    /*void operator<<(ostream& cout,const Person p) {

    }*/
};
// 類外實(shí)現(xiàn)左移運(yùn)算符重載
ostream& operator<<(ostream& o, const Person& p) {
    o << p.age << " " << p.money << endl;
    return o;
}
int main() {
    Person a(5, 12), b;
    cout << a << b << endl;
    return 0;
}

因?yàn)槭窃陬愅鈱?shí)現(xiàn)的,所以在類內(nèi)要將其設(shè)置為友元函數(shù)。參數(shù)為兩個(gè),一個(gè)為ostream類型的o,另外一個(gè)為引用類型的對(duì)象。

返回值為引用類型的ostream類型的ostream,因?yàn)槿绻胍M(jìn)行連續(xù)的輸出,就必須讓前一個(gè)的結(jié)果作為第二個(gè)左移運(yùn)算符的第一個(gè)參數(shù),如下圖:

在o<<p.age的返回值為o,然后它與后面就變成了o<<" ",接著又調(diào)用這個(gè)運(yùn)算符,這樣才能完成連續(xù)的輸出。

3.遞增運(yùn)算符重載

遞增分為兩種一種是前++,一種是后++。前++是在使用之前就讓這個(gè)數(shù)加1,返回的是加1之后的結(jié)果。而后++是先返回這個(gè)數(shù)再進(jìn)行加一操作。

前++返回的是引用,后++返回的是值。

#include <iostream>
using namespace std;
class Person {
    int age;
    int money;

public:
    Person() : age(0), money(0) {}
    Person(int val, int val2) : age(val), money(val2) {}
    Person(const Person& other) {
        this->age = other.age;
        this->money = other.money;
    }
    // 前置++的重載  返回引用
    Person& operator++() { // 保證是對(duì)一個(gè)數(shù)進(jìn)行++
        this->age++;
        this->money++;
        return *this;
    }
    // 后置++的重載 加一個(gè)int參數(shù)占位符區(qū)分 返回值
    Person operator++(int) {
        Person temp = *this; // 用一個(gè)局部變量返回++之前的結(jié)果
        this->age++;
        this->money++;
        return temp;
    }
    void print() { cout << age << " " << money; }
};
int main() {
    Person p(5, 12);
    ++p;
    p.print();
    return 0;
}

對(duì)前++和后++都是operator++,所以為了區(qū)分,對(duì)于后++來(lái)說(shuō),需要加一個(gè)參數(shù)占位符來(lái)進(jìn)行區(qū)分。

在返回值這塊,前++返回的是引用,因?yàn)橐氖撬M(jìn)行加一之后的結(jié)果,所以可以直接返回這個(gè)對(duì)象。

而后++則返回的是一個(gè)值,因?yàn)橐祷氐氖撬右恢暗慕Y(jié)果,所以需要一個(gè)局部變量返回++之前的結(jié)果。

4.+=運(yùn)算符重載

#include <iostream>
using namespace std;
class Person {
    int age;
    int money;

public:
    friend ostream& operator<<(ostream& o, const Person& p);
    Person() : age(0), money(0) {}
    Person(int val, int val2) : age(val), money(val2) {}
    Person(const Person& other) {
        this->age = other.age;
        this->money = other.money;
    }
    Person& operator+=(const Person& other) {
        this->age += other.age;
        this->money += other.money;
        return *this;
    }
    void print() { cout << age << " " << money; }
};
ostream& operator<<(ostream& o, const Person& p) {
    o << p.age << " " << p.money << endl;
    return o;
}
int main() {
    Person a(1, 1), b(2, 2), c(3, 3);
    a += b += c;
    cout << a << b << c;
    return 0;
}

+=運(yùn)算符的返回值也是對(duì)象本身,因?yàn)闀?huì)涉及到連續(xù)+=的情況,和左移運(yùn)算符的重載類似。

這樣才能實(shí)現(xiàn)連續(xù)+=的實(shí)現(xiàn)。如上面代碼運(yùn)行的結(jié)果為:

要注意運(yùn)算的順序是從右往左進(jìn)行計(jì)算的。

5.關(guān)系運(yùn)算符和賦值運(yùn)算符重載

#include <iostream>
using namespace std;
class A {
    int num;
    int* p;

public:
    A() : num(0), p(nullptr) {}
    A(int x) : num(x), p(new int(num)) {}
    A(const A& other) { this->num = other.num; }
    bool operator>(const A& other) {
        if (this->num > other.num)
            return 1;
        else
            return 0;
    }
    bool operator==(const A& other) {
        if (this->num == other.num)
            return 1;
        else
            return 0;
    }
    A& operator=(const A& other) {
        if (p)
            delete p; // 釋放原來(lái)的堆區(qū)內(nèi)存,拷貝構(gòu)造沒有這一步
        num = other.num;
        p = new int(num); // 指向新的堆區(qū)內(nèi)存
        return *this;
    }
};
int& fun() {
    int a = 2;
    return a;
}
int main() {
    int x = fun();  // 行
    int& y = fun(); // 不行
    A a(2), b(3), c;
    cout << (a == b);
    a = b =c; // 如果類內(nèi)沒有實(shí)現(xiàn)賦值運(yùn)算符,編譯器則會(huì)提供默認(rèn)的賦值運(yùn)算符,每個(gè)成員變量都會(huì)被賦值,此時(shí)要注意淺拷貝問題
    return 0;
}

關(guān)系運(yùn)算符主要包括==、>、<等,他們的返回值都是bool類型。

賦值運(yùn)算符,就是把other的值給調(diào)用它的對(duì)象,如果不對(duì)賦值運(yùn)算符進(jìn)行重載的話,那么它就會(huì)是簡(jiǎn)單的賦值操作,這里就會(huì)導(dǎo)致淺拷貝的問題(可以看作者前面提到的淺拷貝和深拷貝問題)。所以我們就需要對(duì)它進(jìn)行重載。

在上述代碼中,因?yàn)閜是一塊指向堆區(qū)的指針變量,所以在進(jìn)行賦值運(yùn)算符重載時(shí),需要開辟一塊新的堆區(qū),然后保證兩個(gè)堆區(qū)的內(nèi)容一樣。同時(shí),還有保證之前這個(gè)指針變量p有沒有指向別的堆區(qū)內(nèi)容,如果有,就將它釋放,避免內(nèi)存泄漏。返回值為引用類型,返回對(duì)象本身。

注意:賦值運(yùn)算符也是編譯器給一個(gè)類至少添加的函數(shù)。

總結(jié)

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • C++形參與實(shí)參的區(qū)別實(shí)例解析

    C++形參與實(shí)參的區(qū)別實(shí)例解析

    這篇文章主要介紹了C++形參與實(shí)參的區(qū)別實(shí)例解析,需要的朋友可以參考下
    2014-07-07
  • C語(yǔ)言中用棧+隊(duì)列實(shí)現(xiàn)隊(duì)列中的元素逆置

    C語(yǔ)言中用棧+隊(duì)列實(shí)現(xiàn)隊(duì)列中的元素逆置

    這篇文章主要介紹了C語(yǔ)言中用利用棧和隊(duì)列實(shí)現(xiàn)隊(duì)列中的元素逆置的相關(guān)資料,對(duì)正在學(xué)習(xí)的小伙伴有一定的參考價(jià)值,需要的可以參考一下,希望對(duì)你有所幫助
    2022-02-02
  • C++初階教程之類和對(duì)象

    C++初階教程之類和對(duì)象

    C++是面向?qū)ο缶幊痰?這也是C++與C語(yǔ)言的最大區(qū)別,而類和對(duì)象就是C++面向?qū)ο蟮幕A(chǔ),下面這篇文章主要給大家介紹了關(guān)于C++初階教程之類和對(duì)象的相關(guān)資料,需要的朋友可以參考下
    2022-02-02
  • C/C++通過(guò)IP獲取局域網(wǎng)網(wǎng)卡MAC地址

    C/C++通過(guò)IP獲取局域網(wǎng)網(wǎng)卡MAC地址

    這篇文章主要為大家詳細(xì)介紹了C++如何通過(guò)Win32API函數(shù)SendARP從IP地址獲取局域網(wǎng)內(nèi)網(wǎng)卡的MAC地址,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2025-02-02
  • C語(yǔ)言實(shí)現(xiàn)游戲VIP停車場(chǎng)管理系統(tǒng)

    C語(yǔ)言實(shí)現(xiàn)游戲VIP停車場(chǎng)管理系統(tǒng)

    這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)游戲VIP停車場(chǎng)管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-12-12
  • C++中返回指向函數(shù)的指針示例

    C++中返回指向函數(shù)的指針示例

    int (*ff(int)) (int *,int);表示:ff(int)是一個(gè)函數(shù),帶有一個(gè)int型的形參,該函數(shù)返回int (*) (int *,int),它是一個(gè)指向函數(shù)的指針,所指向的函數(shù)返回int型并帶有兩個(gè)分別是Int*和int型的形參
    2013-09-09
  • C++中的偽隨機(jī)數(shù)

    C++中的偽隨機(jī)數(shù)

    這篇文章主要介紹了C++中的偽隨機(jī)數(shù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-07-07
  • C++中based for循環(huán)的實(shí)現(xiàn)

    C++中based for循環(huán)的實(shí)現(xiàn)

    C++中的范圍for循環(huán)是一種簡(jiǎn)潔的遍歷容器的方法,本文主要介紹了C++中based for循環(huán)的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下
    2025-02-02
  • 詳解C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之棧

    詳解C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之棧

    這篇文章主要為大家介紹了C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之棧,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助
    2022-01-01
  • c++讀取和寫入TXT文件的整理方法

    c++讀取和寫入TXT文件的整理方法

    今天小編就為大家分享一篇c++讀取和寫入TXT文件的整理方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2018-07-07

最新評(píng)論