操作符重載(二)

来源:https://www.cnblogs.com/songhe364826110/archive/2019/09/23/11575277.html
-Advertisement-
Play Games

[TOC] 1. 數組操作符重載 數組操作符重載 通過重載數組操作符,可以使類的對象支持數組的下標訪問 數組操作符只能重載為類的成員函數 重載函數能且僅能使用一個參數,也就是數組下標 可以定義不同參數的多個重載函數 在重載數組操作符時,要記得數組操作符的原生語義——數組訪問和指針運算。 cpp / ...


目錄

1. 數組操作符重載

數組操作符重載

通過重載數組操作符,可以使類的對象支持數組的下標訪問

  • 數組操作符只能重載為類的成員函數
  • 重載函數能且僅能使用一個參數,也就是數組下標
  • 可以定義不同參數的多個重載函數

在重載數組操作符時,要記得數組操作符的原生語義——數組訪問指針運算
a[n] <--> *(a + n) <--> *(n + a) <--> n[a]

#include <iostream>
#include <string>

using namespace std;

class Test
{
    int a[3];
public:
    int &operator [] (int i)
    {
        return a[i];
    }

    int &operator [] (const string &s)
    {
        if( s == "1st" )
        {
            return a[0];
        }
        else if( s == "2nd" )
        {
            return a[1];
        }
        else if( s == "3rd" )
        {
            return a[2];
        }

        return a[0];
    }
};

int main()
{
    Test t;

    for (int i = 0; i < 3; i++)
    {
        t[i] = i;
    }

    for (int i = 0; i < 3; i++)
    {
        cout << t[i] << endl;
    }

    cout << endl;

    cout << t["3rd"] << endl;
    cout << t["2nd"] << endl;
    cout << t["1st"] << endl;

    return 0;
}

數組類IntArray改進

IntArray.h

class IntArray
{
public:
    int &operator [] (int index); //Add
    IntArray &self();             //Add
};

IntArray.cpp

int &IntArray::operator [] (int index)
{
    return m_pointer[index];
}

IntArray &IntArray::self()
{
    return *this;
}

2. 函數操作符重載(函數對象)

  • 函數操作符只能通過類的成員函數重載
  • 可以定義不同參數的多個重載函數
  • 函數操作符重載的本質是使用具體的類對象取代函數,也就是函數對象,函數對象具備函數調用的行為
  • 函數對象用於在工程中取代函數指針
/*
 * 本示例代碼實現以下需求:
 * - 編寫一個函數,可以獲得斐波那契數列每項的值
 * - 每調用一次返回一個值
 * - 函數可根據需要重覆使用
*/

#include <iostream>
#include <string>

using namespace std;

class Fib
{
    int a0;
    int a1;
public:
    Fib()
    {
        a0 = 0;
        a1 = 1;
    }

    Fib(int n)
    {
        a0 = 0;
        a1 = 1;

        for(int i = 2; i <= n; i++)
        {
            int t = a1;

            a1 = a0 + a1;
            a0 = t;
        }
    }

    int operator () ()
    {
        int ret = a1;

        a1 = a0 + a1;
        a0 = ret;

        return ret;
    }
};

int main()
{
    Fib fib;

    for(int i = 0; i < 10; i++)
    {
        cout << fib() << endl;
    }

    cout << endl;

    for(int i = 0; i < 5; i++)
    {
        cout << fib() << endl;
    }

    cout << endl;

    Fib fib2(10);

    for(int i = 0; i < 5; i++)
    {
        cout << fib2() << endl;
    }

    return 0;
}

3. 指針操作符重載與智能指針

指針操作符重載

指針操作符指的是->*

  • 指針操作符只能通過類的成員函數重載
  • 重載函數不能使用參數,也就是說,只能定義一個重載函數

智能指針

  • 利用指針操作符重載,可以實現智能指針
  • 智能指針只能用來指向堆空間中的對象或者變數!!!
  • 智能指針在生命周期結束時會自動釋放堆空間
  • 智能指針的意義在於減少開發人員的記憶體管理工作,最大程度上避免記憶體問題。
/*
 * 實現智能指針類Pointer,要求如下:
 *  - 一片堆空間最多只能由一個指針標識
 *  - 禁止指針運算和指針比較
*/

#include <iostream>
#include <string>

using namespace std;

class Test
{
    int i;
public:
    Test(int i)
    {
        cout << "Test(int i)" << endl;
        this->i = i;
    }
    int value()
    {
        return i;
    }
    ~Test()
    {
        cout << "~Test()" << endl;
    }
};

class Pointer
{
private:
    Test *mp;
public:
    Pointer(Test *p = NULL)
    {
        mp = p;
    }

    Pointer(const Pointer &obj)
    {
        mp = obj.mp;
        const_cast<Pointer &>(obj).mp = NULL;
    }

    Pointer &operator = (const Pointer &obj)
    {
        if (this != &obj)
        {
            delete mp;
            mp = obj.mp;
            const_cast<Pointer &>(obj).mp = NULL;
        }

        return *this;
    }

    Test *operator -> ()
    {
        return mp;
    }

    Test &operator * ()
    {
        return *mp;
    }

    bool isNull()
    {
        return (mp == NULL);
    }

    ~Pointer()
    {
        delete mp;
    }
};

int main()
{
    Pointer p1 = new Test(0);
    Pointer p2 = p1;

    cout << p1.isNull() << endl;
    cout << p2->value() << endl;

    Pointer p3;
    p3 = p2;

    cout << p2.isNull() << endl;
    cout << p3->value() << endl;

    return 0;
}

4. 前置、後置操作符重載

重載實現

前置、後置操作符指的是++--,我們以++為例進行講解,--和++是一樣的

  • 全局函數成員函數均可進行重載
  • 重載前置++操作符不需要參數
  • 重載後置++操作符需要一個int類型的占位參數
#include <iostream>
#include <string>

using namespace std;

class Test
{
    int mValue;
public:
    Test(int i)
    {
        mValue = i;
    }

    int value()
    {
        return mValue;
    }

    Test &operator ++ ()
    {
        ++mValue;

        return *this;
    }

    Test operator ++ (int)
    {
        Test ret(mValue);

        mValue++;

        return ret;
    }
};

int main()
{
    Test t1(0);
    Test t2(0);

    Test t = t1++;

    cout << t.value() << endl;
    cout << t1.value() << endl;

    t = ++t2;

    cout << t.value() << endl;
    cout << t2.value() << endl;

    return 0;
}

前置、後置重載的區別

  • 對於基礎類型的變數,前置++和後置++的效率基本相同,沒有什麼區別
  • 對於類類型的對象,前置++的效率高於後置++,因為後置++重載會調用構造與析構函數
  • 在工程中儘量使用前置++重載以提高程式效率

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 橋接模式(Bridge): 橋接是用於把抽象化與實現化解耦,使得兩者可以獨立變化。 橋接模式的角色: 1)抽象化角色(Abstraction):它是用於定義抽象介面,通常是抽象類而不是介面,其中定義了一個Implementor(實現介面)類型的對象並可以維護該對象,它與Implementor之間具有 ...
  • 1. 下載安裝包 1 2 3 https://www.python.org/ftp/python/2.7.14/python-2.7.14.amd64.msi # 2.7安裝包 https://www.python.org/ftp/python/3.6.4/python-3.6.4-amd64.ex ...
  • ​一、生成器 1.定義(generator):一邊迴圈一邊計算下一個元素的機制/演算法 2.滿三個條件 (1)每次調用都能產生出for迴圈需要的下一個元素 (2)如果達到最後一個後,能夠爆出StopIteration異常 (3)可以被next函數調用 3.如何生成一個生成器 (1)直接使用 (2)如果 ...
  • # 1.在伺服器上 tomcat 的 bin目錄下找到並打開 catalina.sh 在文件中搜索: ``` JPDA_ADDRESS= ``` 找一個伺服器上沒有被使用的埠,填入,如50005,保存並退出。 > 如何知道某埠有沒有被占用? > 命令: > ``` > lsof -i:50005 ...
  • idea搭建spring源碼閱讀環境 安裝gradle Github下載Spring源碼 新建學習spring源碼的項目 idea搭建spring源碼閱讀環境 安裝gradle Github下載Spring源碼 新建學習spring源碼的項目 安裝gradle Github下載Spring源碼 新建 ...
  • 2019-09-23-23:48:00 今日所學的內容有: ...
  • 一、預設配置文件 二、指定配置文件 三、使用profile指定配置 ...
  • 在上篇文章: "SpringBoot源碼解析:創建SpringApplication對象實例" 中,我們詳細描述了SpringApplication對象實例的創建過程,本篇文章繼續看 方法的執行邏輯吧 1. 第一行使用了 來記錄開始時間 2. 設置了 環境變數,在網上瞭解了一下這個變數的相關信息 H ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...