C++ STL迭代器原理和實現

来源:https://www.cnblogs.com/wengle520/archive/2020/03/14/12492708.html
-Advertisement-
Play Games

1. 迭代器簡介 為了提高C++編程的效率,STL(Standard Template Library)中提供了許多容器,包括vector、list、map、set等。然而有些容器(vector)可以通過下標索引的方式訪問容器裡面的數據,但是大部分的容器(list、map、set)不能使用這種方式訪 ...


1. 迭代器簡介

為了提高C++編程的效率,STL(Standard Template Library)中提供了許多容器,包括vector、list、map、set等。然而有些容器(vector)可以通過下標索引的方式訪問容器裡面的數據,但是大部分的容器(list、map、set)不能使用這種方式訪問容器中的元素。為了統一訪問不同容器時的訪問方式,STL為每種容器在實現的時候設計了一個內嵌的iterator類,不同的容器有自己專屬的迭代器(專屬迭代器負責實現對應容器訪問元素的具體細節),使用迭代器來訪問容器中的數據。除此之外,通過迭代器可以將容器和通用演算法結合在一起,只要給予演算法不同的迭代器,就可以對不同容器執行相同的操作,例如find查找函數(因為迭代器提供了統一的訪問方式,這是使用迭代器帶來的好處)。迭代器對一些基本操作如*、->、++、==、!=、=進行了重載,使其具有了遍歷複雜數據結構的能力,其遍歷機制取決於所遍歷的容器,所有迭代器的使用和指針的使用非常相似。通過begin,end函數獲取容器的頭部和尾部迭代器,end迭代器不包含在容器之內,當begin和end返回的迭代器相同時表示容器為空。

STL主要由 容器、迭代器、演算法、函數對象、和記憶體分配器 五大部分構成。

2. 迭代器的實現原理

首先,看看STL中迭代器的實現思路:
STL中迭代器的實現思路
從上圖中可以看出,STL通過類型別名的方式實現了對外統一;在不同的容器中類型別名的真實迭代器類型是不一樣的,而且真實迭代器類型對於++、--、*、->等基本操作的實現方式也是不同的。(PS:迭代器很好地詮釋了介面與實現分離的意義)

既然我們已經知道了迭代器的實現思路,現在如果讓我們自己設計一個list容器的簡單迭代器,應該如何實現呢?

  1. list類需要有操作迭代器的方法
    1. begin/end
    2. insert/erase/emplace
  2. list類有一個內部類list_iterator
    1. 有一個成員變數ptr指向list容器中的某個元素
    2. iterator負責重載++、--、*、->等基本操作
  3. list類定義內部類list_iterator的類型別名

以上就是實現一個list容器的簡單迭代器需要考慮的具體細節。

3. 迭代器的簡單實現

my_list.h(重要部分有註釋說明

//
// Created by wengle on 2020-03-14.
//

#ifndef CPP_PRIMER_MY_LIST_H
#define CPP_PRIMER_MY_LIST_H

#include <iostream>

template<typename T>
class node {
public:
    T value;
    node *next;
    node() : next(nullptr) {}
    node(T val, node *p = nullptr) : value(val), next(p) {}
};

template<typename T>
class my_list {
private:
    node<T> *head;
    node<T> *tail;
    int size;

private:
    class list_iterator {
    private:
        node<T> *ptr; //指向list容器中的某個元素的指針

    public:
        list_iterator(node<T> *p = nullptr) : ptr(p) {}
        
        //重載++、--、*、->等基本操作
        //返回引用,方便通過*it來修改對象
        T &operator*() const {
            return ptr->value;
        }

        node<T> *operator->() const {
            return ptr;
        }

        list_iterator &operator++() {
            ptr = ptr->next;
            return *this;
        }

        list_iterator operator++(int) {
            node<T> *tmp = ptr;
            // this 是指向list_iterator的常量指針,因此*this就是list_iterator對象,前置++已經被重載過
            ++(*this);
            return list_iterator(tmp);
        }

        bool operator==(const list_iterator &t) const {
            return t.ptr == this->ptr;
        }

        bool operator!=(const list_iterator &t) const {
            return t.ptr != this->ptr;
        }
    };

public:
    typedef list_iterator iterator; //類型別名
    my_list() {
        head = nullptr;
        tail = nullptr;
        size = 0;
    }

    //從鏈表尾部插入元素
    void push_back(const T &value) {
        if (head == nullptr) {
            head = new node<T>(value);
            tail = head;
        } else {
            tail->next = new node<T>(value);
            tail = tail->next;
        }
        size++;
    }

    //列印鏈表元素
    void print(std::ostream &os = std::cout) const {
        for (node<T> *ptr = head; ptr != tail->next; ptr = ptr->next)
            os << ptr->value << std::endl;
    }

public:
    //操作迭代器的方法
    //返回鏈表頭部指針
    iterator begin() const {
        return list_iterator(head);
    }

    //返回鏈表尾部指針
    iterator end() const {
        return list_iterator(tail->next);
    }

    //其它成員函數 insert/erase/emplace
};

#endif //CPP_PRIMER_MY_LIST_H

test.cpp

//
// Created by wengle on 2020-03-14.
//

#include <string>
#include "my_list.h"

struct student {
    std::string name;
    int age;

    student(std::string n, int a) : name(n), age(a) {}

    //重載輸出操作符
    friend std::ostream &operator<<(std::ostream &os, const student &stu) {
        os << stu.name << " " << stu.age;
        return os;
    }
};

int main() {
    my_list<student> l;
    l.push_back(student("bob", 1)); //臨時量作為實參傳遞給push_back方法
    l.push_back(student("allen", 2));
    l.push_back(student("anna", 3));
    l.print();

    for (my_list<student>::iterator it = l.begin(); it != l.end(); it++) {
        std::cout << *it << std::endl;
        *it = student("wengle", 18);
    }
    return 0;
}

4. 迭代器失效

// inserting into a vector
#include <iostream>
#include <vector>

int main ()
{
  std::vector<int> myvector (3,100);
  std::vector<int>::iterator it;

  it = myvector.begin();
  it = myvector.insert ( it , 200 );

  myvector.insert (it,200,300);
  //it = myvector.insert (it,200,300);
  myvector.insert (it,5,500); //當程式執行到這裡時,大概率會crash
  for (std::vector<int>::iterator it2=myvector.begin(); it2<myvector.end(); it2++)
    std::cout << ' ' << *it2;
  std::cout << '\n';

  return 0;
}

上面的代碼很好地展示了什麼是迭代器失效?迭代器失效會導致什麼樣的問題?
當執行完myvector.insert (it,200,300);這條語句後,實際上myvector已經申請了一塊新的記憶體空間來存放之前已保存的數據本次要插入的數據,由於it迭代器內部的指針還是指向舊記憶體空間的元素,一旦舊記憶體空間被釋放,當執行myvector.insert (it,5,500);時就會crash(PS:因為你通過iterator的指針正在操作一塊已經被釋放的記憶體,大多數情況下都會crash)。迭代器失效就是指:迭代器內部的指針沒有及時更新,依然指向舊記憶體空間的元素。

insert源碼

上圖展示了STL源碼中vector容器insert方法的實現方式。當插入的元素個數超過當前容器的剩餘容量時,就會導致迭代器失效。這也是測試代碼中myvector.insert (it,200,300);插入200個元素的原因,為了模擬超過當前容器剩餘容量的場景,如果你的測試環境沒有crash,可以將插入元素設置的更多一些。


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

-Advertisement-
Play Games
更多相關文章
  • Dubbo admin管理控制台目前還沒有正式發佈,但是源碼已托管在github上,我們可以自行下載使用; 目前的管理控制台已經發佈0.1版本,結構上採取了前後端分離的方式,前端使用Vue和Vuetify分別作為Javascript框架和UI框架,後端採用Spring Boot框架。既可以按照標準的 ...
  • 1、Spring 1.x時代 在Spring 1.x時代,都是通過XML文件配置Bean。隨著項目的不斷擴大,需要將Bean的定義配置分放到不同的XML配置文件中。開發的時候需要頻繁的在java類和XML配置文件中切換。 2、Spring 2.x時代 隨著 JDK 1.5帶來的註解支持,Spring ...
  • 資源限制 時間限制:1.0s 記憶體限制:256.0MB 問題描述 利用字母可以組成一些美麗的圖形,下麵給出了一個例子: ABCDEFG BABCDEF CBABCDE DCBABCD EDCBABC 這是一個5行7列的圖形,請找出這個圖形的規律,並輸出一個n行m列的圖形。 輸入格式 輸入一行,包含兩 ...
  • [TOC] pandas對象有一個常用數學,統計學方法的集合。大部分屬於歸納或彙總統計。這些方法從DataFrame的行或列中抽取一個Series或一系列的值。 pandas的描述性統計的方法和NumPy的方法相比,內建了處理缺失值的功能,很好地針對於每一個我們需要處理的數據。 一:一些基本方法 1 ...
  • 作者:Moon Light Dream 出處:https://www.cnblogs.com/Moon Light Dream/ 轉載:歡迎轉載,但未經作者同意,必須保留此段聲明;必須在文章中給出原文連接;否則必究法律責任 什麼是go cache KV存儲引擎有很多,常用的如redis,rocksd ...
  • 關於這篇博文 關於面向對象的話題,是一個十分重要的話題,由於博主是做java出身的,在java 中,由於java的嚴謹性,面向對象可以搞死你(/很難很難),在python中,所有的 一切都TM怎麼簡單,所以,這放出了詳細的python面向對象階段的學習筆記,供你 去翻閱,實例代碼也在裡面。 項目地址 ...
  • @Param()註解 引用類型不需要加 如果只有一個基本類型的,可以忽略 在 sql 中引用的就是在註解中設定的欄位名 高級結果映射 多對一: 一對多 動態 sql if choose (when, otherwise) trim (where, set) foreach 實體類 if 如果上面的 ...
  • 1. SpringMVC的Controller實現方式 SpringMVC實現Controller的方式主要有控制器實現方式與全註解實現方式,其中全註解實現方式是當前項目中比較常用的一種方式。 1.1.控制器實現方式 1.1.1. 實現Controller介面 創建一個類實現Controller介面 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...