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

C++模擬實現(xiàn)List迭代器詳解

 更新時間:2022年04月15日 12:15:19   作者:m0_52012656  
list不同于其他容器,他是一個鏈表,物理地址并不連續(xù)。所以在實現(xiàn)list類的迭代器的時候,需要將迭代器單獨封裝到一個類里,因為需要重載很多操作符來跟其他容器的迭代器使用達成一致

概念

迭代器是一種抽象的設(shè)計概念,其定義為:提供一種方法,使他能夠按順序遍歷某個聚合體(容器)所包含的所有元素,但又不需要暴露該容器的內(nèi)部表現(xiàn)方式。

迭代器是一種行為類似智能指針的對象, 而指針最常見的行為就是內(nèi) 容提領(lǐng)和成員 訪問。 因此迭代器最重要的行為就是對operator*和operator->進行重載。

STL的中心思想在于: 將數(shù)據(jù)容器和算法分開, 彼此獨立設(shè)計, 最后再以一貼膠合劑( iterator) 將它們撮合在一起。STL的迭代器是一個可遍歷STL容器全部或者部分?jǐn)?shù)據(jù)

迭代器使用

我們可以使用迭代器訪問修改鏈表元素

list<int>  lt;
list<int>::iterator it=lt.begin();
while(it!=lt.end())
{
    *it+=2;
    cout<<*it<<" ";
    it++;
}

2.我們有些函數(shù)接口需要傳迭代器,例如:

template <class InputIterator, class T>
   InputIterator find (InputIterator first, InputIterator last, const T& val);
?
template <class ForwardIterator, class T>
  void replace (ForwardIterator first, ForwardIterator last,
                const T& old_value, const T& new_value);

迭代器模擬實現(xiàn)

迭代器的大體結(jié)構(gòu)

//鏈表節(jié)點
template<class T>
struct ListNode {
    ListNode<T>* _next;
    ListNode<T>* _prev;
    T _data;
    //構(gòu)造節(jié)點值
    ListNode(const T& data = T())
        :_next(nullptr)
        ,_prev(nullptr)
        ,_data(data)
    {}
};
?
///迭代器
//T為list數(shù)據(jù)類型,Ref為T&,Ptr為T*
template<class T,class Ref,class Ptr>
struct __list_iterator
{
    typedef ListNode<T> Node;
    typedef __list_iterator<T,Ref,Ptr> self;
    Node* _node;//節(jié)點指針
    
    //接下來實現(xiàn)的函數(shù)都是在這個位置
};

構(gòu)造函數(shù)

一般都會傳過來一個節(jié)點地址

__list_iterator(Node* x)
    :_node(x)
{ }

注意迭代器的拷貝構(gòu)造、賦值重載以及析構(gòu)函數(shù)不需要我們自己實現(xiàn),編譯器實現(xiàn)的完全夠用。

  • 拷貝構(gòu)造與賦值重載:因為list迭代器本身就是一個自定義類型的指針,都是地址的拷貝與賦予。所以淺拷貝就滿足使用。
  • 析構(gòu)函數(shù):因為list迭代器是借助節(jié)點指針訪問修改鏈表,節(jié)點是鏈表的,不需要迭代器釋放。

解引用重載

解引用重載(*)

解引用本質(zhì)是根據(jù)地址拿到在這個地址的有效數(shù)據(jù)

Ref operator*()
{
    return _node->_data;
}

重載

->重載

->本質(zhì)是拿到所求數(shù)據(jù)的地址

Ptr operator->()
{
    return &_node->_data;
}

自增實現(xiàn)

前置++

++后迭代器指向當(dāng)前位置的下一個位置,返回指向下一個位置的迭代器

self& operator++()
{
    _node=_node->_next;
    return *this;
}

后置++

++后迭代器指向當(dāng)前位置的下一個位置,返回指向之前位置的迭代器,要使用一個臨時變量保存++之前的this指針,然后后移_node,返回臨時變量。

//這塊一定要使用占位符,防止與前置++重命名。
self& operator++(int)
{
    __list_iterator<T> tmp(*this);
    _node=_node->_next;
    return tmp;
}

自減實現(xiàn)

與++基本一樣,不做解釋。

前置--

self& operator--()
{
    _node=_node->_prev;
    return *this;
}

后置--

self& operator--(int)
{
    __list_iterator<T> tmp(*this);
    _node=_node->_prev;
    return tmp;
}

運算符重載

bool operator!=(const self& it)const
{
    return _node!=it._node;
}
 
bool operator==(const self& it)const
{
    return _node==it._node;
}

迭代器失效

以vector為例,當(dāng)我們插入一個元素時它的預(yù)分配空間不夠時,它會重新申請一段新空間,將原空間上的元素 復(fù)制到新的空間上去,然后再把新加入的元素放到新空間的尾部,以滿足vector元素要求連續(xù)存儲的目的。而后原空間會被系統(tǒng)撤銷或征做他用,于是指向原 空間的迭代器就成了類似于“野指針”一樣的東西,指向了一片非法區(qū)域。如果使用了這樣的迭代器會導(dǎo)致嚴(yán)重的運行時錯誤就變得很自然了。這也是許多書上敘 述vector在insert操作后“可能導(dǎo)致所有迭代器實效”的原因。

但是想到這里我不禁想到vector的erase操作的敘述是“會導(dǎo)致指向刪除元 素和刪除元素之后的迭代器失效” ,這里的刪除元素不一定不成功,但一定存在迭代器失效。例:

vector<int> v;//{1,2,3,4,5}
vector<int>::iterator it=v.begin();
while(it!=v.end())
{
    if(*it%2==0)
    {
        v.erase(it);
    }
    it++;
}

所以要避免這種情況,改進代碼

vector<int> v;//{1,2,3,4,5}
vector<int>::iterator it=v.begin();
while(it!=v.end())
{
    if(*it%2==0)
    {
        v.erase(it);
    }
    else
        it++;
}

list迭代器失效

list<int> l1;
list<int>::iterator it=l1.begin();
while(it!=l1.end())
{
    if(*it%2==0)
    {
        l1.erase(it);
    }
    else
        ++it;
}

改進代碼

list<int> l1;
list<int>::iterator it=l1.begin();
while(it!=l1.end())
{
    if(*it%2==0)
    {
        it=l1.erase(it);
    }
    else
        ++it;
}

歸納迭代器失效的類型

(1)由于容器元素整體“遷移”導(dǎo)致存放原容器元素的空間不再有效,從而使得指向原空間的迭代器失效。

(2)由于刪除元素使得某些元素次序發(fā)生變化使得原本指向某元素的迭代器不再指向希望指向的元素

模擬List

具體下一章講

 template<class T>
    class list
    {
        typedef ListNode<T> Node;
    public:
        typedef __list_iterator<T, T&, T*> iterator;
        typedef __list_iterator<T, const T&, const T*> const_iterator;
        iterator begin()
        {
            return iterator(_head->_next);
        }
?
        iterator end()
        {
            return iterator(_head);
        }
?
        const_iterator begin()const
        {
            return const_iterator(_head->_next);
        }
?
        const_iterator end()const
        {
            return const_iterator(_head);
        }
?
        list()
        {
            _head = new Node();
            _head->_next = _head;
            _head->_prev = _head;
        }
        void push_back(const T& x)
        {
            Node* tail = _head->_prev;
            Node* newnode = new Node(x);
            tail->_next = newnode;
            newnode->_prev = tail;
            newnode->_next = _head;
            _head->_prev = newnode;
        }
?
        void insert(iterator pos, const T& x)
        {
?
        }
        void erase(iterator pos)
        {
?
        }
    private:
        Node* _head;
    };

到此這篇關(guān)于C++模擬實現(xiàn)List迭代器詳解的文章就介紹到這了,更多相關(guān)C++ List迭代器內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C++基本算法思想之遞推算法思想

    C++基本算法思想之遞推算法思想

    遞推算法需要用戶知道答案和問題之間的邏輯關(guān)系。在許多數(shù)學(xué)問題中,都有明確的計算公式可以遵循,因此可以采用遞推算法來實現(xiàn)
    2013-10-10
  • C語言自定義類型的保姆級講解

    C語言自定義類型的保姆級講解

    這篇文章主要給大家介紹了關(guān)于C語言自定義類型的保姆級講解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-03-03
  • 數(shù)據(jù)結(jié)構(gòu)順序表操作示例

    數(shù)據(jù)結(jié)構(gòu)順序表操作示例

    這篇文章主要介紹了數(shù)據(jù)結(jié)構(gòu)順序表操作示例,其中有在第I個元素前插入數(shù)據(jù)x,元素從0開始計數(shù)、刪除第i個元素,元素從0開始計數(shù)的方法,需要的朋友可以參考下
    2014-03-03
  • 老生常談C++的單例模式與線程安全單例模式(懶漢/餓漢)

    老生常談C++的單例模式與線程安全單例模式(懶漢/餓漢)

    下面小編就為大家?guī)硪黄仙U凜++的單例模式與線程安全單例模式(懶漢/餓漢)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-12-12
  • Qt網(wǎng)絡(luò)編程之TCP通信及常見問題

    Qt網(wǎng)絡(luò)編程之TCP通信及常見問題

    這篇文章主要為大家詳細(xì)介紹了Qt網(wǎng)絡(luò)編程之TCP通信及常見問題,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-08-08
  • C++多線程強制終止詳細(xì)

    C++多線程強制終止詳細(xì)

    這篇文章主要介紹了C++多線程強制終止, 實際上,沒有任何語言或操作系統(tǒng)可以為你提供異步突然終止線程的便利,且不會警告你不要使用它們。但是下面我們再來簡單看看相關(guān)內(nèi)容吧
    2021-09-09
  • 判斷指定的進程或程序是否存在方法小結(jié)(vc等)

    判斷指定的進程或程序是否存在方法小結(jié)(vc等)

    VC判斷進程是否存在?比如我想知道記事本是否運行,要用到哪些函數(shù)等實例,需要的朋友可以參考下
    2013-01-01
  • vscode使用官方C/C++插件無法進行代碼格式化問題

    vscode使用官方C/C++插件無法進行代碼格式化問題

    這篇文章主要介紹了vscode使用官方C/C++插件無法進行代碼格式化問題,本文通過截圖實例代碼相結(jié)合給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-04-04
  • C 語言基礎(chǔ)教程(我的C之旅開始了)[六]

    C 語言基礎(chǔ)教程(我的C之旅開始了)[六]

    C 語言基礎(chǔ)教程(我的C之旅開始了)[六]...
    2007-02-02
  • C語言實現(xiàn)簡單的掃雷游戲

    C語言實現(xiàn)簡單的掃雷游戲

    這篇文章主要為大家詳細(xì)介紹了C語言實現(xiàn)簡單的掃雷游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-12-12

最新評論