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

C++迭代器失效問題及解決

 更新時間:2023年02月10日 16:46:57   作者:倒地不起的土豆  
這篇文章主要介紹了C++迭代器失效問題及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教

什么是迭代器

迭代器不是指針,是類模板,表現(xiàn)的像指針。他只是模擬了指針的一些功能,通過重載了指針的一些操作符,->,*,++ --等封裝了指針,是一個“可遍歷STL( Standard Template Library)容器內(nèi)全部或部分元素”的對象, 本質(zhì)是封裝了原生指針,他可以根據(jù)不同類型的數(shù)據(jù)結(jié)構(gòu)來實現(xiàn)不同的++,–,*等操作;

迭代器只能指向容器,而指針還可以指向函數(shù),

迭代器返回的是對象引用而不是對象的值,所以cout只能輸出迭代器使用*取值后的值而不能直接輸出其自身。

迭代器失效

迭代器失效就是迭代器底層對應(yīng)指針?biāo)赶虻目臻g倍銷毀了,導(dǎo)致使用了一塊已經(jīng)被釋放了的空間。

容器操作可能使迭代器失效,一個失效的迭代器將不再表示任何元素。使用失效的迭代器是一個嚴(yán)重的程序設(shè)計錯誤,很可能產(chǎn)生與使用未初始化化指針一樣的問題。

順序容器迭代器失效

添加操作

容器是vector或者string的情況下,在向容器添加元素后,如果容器擴(kuò)容,即容器的存儲空間重新分配,則指向容器的迭代器失效。

以vector為例,當(dāng)我們插入一個元素時它的預(yù)分配空間不夠時,它會重新申請一段新空間,將原空間上的元素復(fù)制到新的空間上去,以滿足vector元素要求連續(xù)存儲的目的,而原空間會被系統(tǒng)撤銷或征做他用,于是指向原空間的迭代器就成了類似于“空懸指針”一樣的東西,指向了一片非法區(qū)域,從而使得指向原空間的迭代器失效。

int main()
{
	vector<int>v1;
	vector<int>::iterator it1 = v1.begin();
	for (int i = 0;i < 10;i++) {
		v1.push_back(i);
		cout << "Size:" << v1.size() << "  Capacity:" << v1.capacity() << "  address:" << &v1[0] << endl;
	}
	return 0;
}

容器是vector或者string的情況下,在向容器添加元素后,如果容器未擴(kuò)容,則指向插入位置之前的元素的迭代器有效,但指向插入位置之后元素的迭代器將會失效,這是因為插入位置后的元素次序發(fā)生變化使得原本指向某元素的迭代器不再指向希望指向的元素。

對于push_back()操作,end操作返回的迭代器肯定失效

int main()
{
	vector<int>v1 = { 0,1,2,3 };
	v1.reserve(10);//分配至少能容納10個元素的內(nèi)存空間
	vector<int>::iterator it1 = v1.begin();
	vector<int>::iterator it2 = v1.begin() + 1;
	vector<int>::iterator it3 = v1.begin() + 2;
	v1.insert(v1.begin() + 2, 4);
	while (it1 <= it2) {	//輸出0 1
		cout << *it1 << endl;
		it1++;
	}
	cout << *it3 << endl; //此時該迭代器已失效,指向非法的區(qū)域
	return 0;
}

容器是deque的情況下,插入到任何位置都會導(dǎo)致迭代器。

插入前迭代器處于正常狀態(tài)

插入后所有迭代器失效

容器是list和forward_list的情況下,插入到任何位置,指向容器的迭代器仍然有效,

刪除操作

刪除操作會導(dǎo)致元素次序發(fā)送改變而導(dǎo)致迭代器失效

容器是vector或者string的情況下,在刪除容器元素后,指向被刪元素之前元素的迭代器有效,被刪元素之后的迭代器失效

注意:當(dāng)我們刪除元素時,尾后迭代器總是會失效

容器是deque的情況下,如果刪除的是首尾元素,則只有首尾迭代器失效,如果刪除的是其他元素,則所有迭代器失效。

容器是list和forward_list的情況下,刪除任何元素,僅僅會使被刪除元素 的迭代器失效,這是因為 list 之類的容器,使用了鏈表來實現(xiàn),插入、刪除一個結(jié)點(diǎn)不會對其他結(jié)點(diǎn)造成影響。

關(guān)聯(lián)容器迭代器失效

對于關(guān)聯(lián)容器map、multimap、set、multiset,底層是使用紅黑樹實現(xiàn)的,所以刪除某個元素,僅僅會刪除元素的迭代器失效。

插入、刪除一個結(jié)點(diǎn)不會對其他結(jié)點(diǎn)造成影響。

swap()操作迭代器為什么不失效

因為swap操作不會對容器的任何元素進(jìn)行拷貝、刪除或插入操作,也意味著元素不會被移動,這也就是迭代器為什么不失效的原因,因為指向容器的迭代器swap操作之后,仍指向swap操作之前所指向的那些元素,只不過這些元素已經(jīng)屬于不同的容器了

注意:對于string容器調(diào)用swap操作會導(dǎo)致迭代器失效,而swap兩個array會真正交換它門的元素

int main()
{
	vector<int>v1 = { 0,1,2,3 };
	vector<int>v2 = { 4,5,6,7};
	vector<int>::iterator it1 = v1.begin();
	vector<int>::iterator it2 = v2.begin();
	vector<int>::iterator tmp1 = v1.begin();
	vector<int>::iterator tmp2 = v2.begin();
	cout << "v1: ";
	for (const auto&n : v1) {
		cout << n << " ";
	}
	cout << endl;
	cout << "v2: ";
	for (const auto&n : v2) {
		cout << n << " ";
	}
	cout << endl;
	while (it1 != v1.end()) {
		cout << *it1 << " ";
		it1++;
	}
	cout << endl;
	while (it2 != v2.end()) {
		cout << *it2 << " ";
		it2++;
	}
	cout << endl;
	swap(v1, v2); //交換容器v1,v2;
	it1 = tmp1;
	it2 = tmp2;
	cout << "v1: ";
	for (const auto&n : v1) {
		cout << n << " ";
	}
	cout << endl;
	cout << "v2: ";
	for (const auto&n : v2) {
		cout << n << " ";
	}
	cout << endl;
	while (it1 != v2.end()) {
		cout << *it1 << " ";
		it1++;
	}
	cout << endl;
	while (it2 != v1.end()) {
		cout << *it2 << " ";
		it2++;
	}

	cout << endl;
	return 0;
}

輸出:

可以看到swap操作后,迭代器未失效,且指向相同的元素,只不過這些元素已經(jīng)屬于不同的容器了;

注意事項

不用在循環(huán)之前保存end返回的迭代器,一直當(dāng)做容器末尾使用,因為如果在循環(huán)中添加/刪除 vector或string的元素后,或在deque中首元素之外任何位置添加/刪除元素后,原來end返回的迭代器總會失效。

因此在循環(huán)中我們必須反復(fù)調(diào)用end()來獲取迭代器

對于刪除元素導(dǎo)致的迭代器失效解決方法

方法1.erase§時,尾后遞增當(dāng)前迭代器,即erase(p++);

方法2.由于erase()可以返回一個指向被刪除元素之后元素的迭代器,所以也可以直接把erase§的返回值賦給當(dāng)前迭代器,即p=erase§

注意:對于vector、string這種在內(nèi)存中連續(xù)存儲的只能使用方法1,因為刪除元素后,后面元素的迭代器也失效了

對于關(guān)聯(lián)容器,也只能使用方法1,因為它們的erase操作返回值為void。

總結(jié)

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

相關(guān)文章

  • C語言實現(xiàn)水波紋效果

    C語言實現(xiàn)水波紋效果

    這篇文章主要為大家詳細(xì)介紹了C語言實現(xiàn)水波紋效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-02-02
  • Windows下sentry接入C/C++程序的詳細(xì)過程

    Windows下sentry接入C/C++程序的詳細(xì)過程

    sentry作為一個開源的軟件,發(fā)展至今,已經(jīng)非常成熟。它支持的平臺眾多,甚至于針對不同的工作者(后臺、前端、客戶端)都有相應(yīng)的內(nèi)容,這篇文章主要介紹了Windows下sentry接入C/C++程序,需要的朋友可以參考下
    2022-09-09
  • C語言實現(xiàn)掃雷游戲詳細(xì)流程

    C語言實現(xiàn)掃雷游戲詳細(xì)流程

    windows自帶的游戲《掃雷》是陪伴了無數(shù)人的經(jīng)典游戲,本文將利用C語言實現(xiàn)這一經(jīng)典的游戲,文中的示例代碼講解詳細(xì),感興趣的可以學(xué)習(xí)一下
    2022-05-05
  • VSCode搭建C/C++編譯環(huán)境的詳細(xì)教程

    VSCode搭建C/C++編譯環(huán)境的詳細(xì)教程

    Visual Studio Code是一款免費(fèi)開源的現(xiàn)代化輕量級代碼編輯器,支持幾乎所有主流的開發(fā)語言的語法高亮、智能代碼補(bǔ)全、自定義熱鍵、括號匹配、代碼片段、代碼對比 Diff、GIT 等特性,這篇文章主要介紹了VSCode搭建C/C++編譯環(huán)境,需要的朋友可以參考下
    2020-05-05
  • Qt 自定義分頁控件的實現(xiàn)

    Qt 自定義分頁控件的實現(xiàn)

    在應(yīng)用程序開發(fā)時經(jīng)常會遇到數(shù)據(jù)分頁的需求,每一頁展示特定數(shù)量的數(shù)據(jù),通過點(diǎn)擊按鈕翻頁或者輸入頁碼跳轉(zhuǎn)到指定頁,本文就來介紹一下Qt 自定義分頁控件的實現(xiàn),感興趣的可以了解一下
    2023-11-11
  • C++中的模板類繼承和成員訪問問題

    C++中的模板類繼承和成員訪問問題

    這篇文章主要介紹了C++中的模板類繼承和成員訪問問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-11-11
  • 線程池的原理與實現(xiàn)詳解

    線程池的原理與實現(xiàn)詳解

    下面利用C語言來實現(xiàn)一個簡單的線程池,為了使得這個線程池庫使用起來更加方便,特在C實現(xiàn)中加入了一些OO的思想,與Objective-C不同,它僅僅是使用了struct來模擬了c++中的類,其實這種方式在linux內(nèi)核中大量可見
    2013-09-09
  • VC枚舉串口端口應(yīng)用

    VC枚舉串口端口應(yīng)用

    這篇文章主要介紹了VC枚舉串口端口應(yīng)用,羅列了常見的一些串口端口的應(yīng)用實例,需要的朋友可以參考下
    2014-10-10
  • VSCode遠(yuǎn)程代碼開發(fā)及DNS隧道端口轉(zhuǎn)發(fā)實現(xiàn)遠(yuǎn)程辦公代碼

    VSCode遠(yuǎn)程代碼開發(fā)及DNS隧道端口轉(zhuǎn)發(fā)實現(xiàn)遠(yuǎn)程辦公代碼

    這篇文章主要介紹了VSCode遠(yuǎn)程代碼開發(fā)及DNS隧道端口轉(zhuǎn)發(fā)實現(xiàn)遠(yuǎn)程辦公,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-04-04
  • 詳解C語言中動態(tài)內(nèi)存管理及柔性數(shù)組的使用

    詳解C語言中動態(tài)內(nèi)存管理及柔性數(shù)組的使用

    這篇文章主要為大家詳細(xì)介紹一下C語言中動態(tài)內(nèi)存管理以及柔性數(shù)組的使用方法,文中的示例代碼講解詳細(xì),對我們學(xué)習(xí)C語言有一定的幫助,需要的可以參考一下
    2022-07-07

最新評論