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

C++模擬實(shí)現(xiàn)vector示例代碼圖文講解

 更新時(shí)間:2023年02月27日 10:12:30   作者:北冥有魚(yú)丶丶  
這篇文章主要介紹了C++容器Vector的模擬實(shí)現(xiàn),Vector是一個(gè)能夠存放任意類(lèi)型的動(dòng)態(tài)數(shù)組,有點(diǎn)類(lèi)似數(shù)組,是一個(gè)連續(xù)地址空間,下文更多詳細(xì)內(nèi)容的介紹,需要的小伙伴可以參考一下

vector的模擬實(shí)現(xiàn)

#include <iostream>
using namespace std;
#include <assert.h>
namespace myVector
{
	template<class T>
	class vector
	{
	public:
		// Vector的迭代器是一個(gè)原生指針
		typedef T* iterator;
		typedef const T* const_iterator;
		///
		// 構(gòu)造和銷(xiāo)毀
		vector()
			: _start(nullptr)
			, _finish(nullptr)
			, _endOfStorage(nullptr)
		{}
		vector(size_t n, const T& value = T())
			: _start(nullptr)
			, _finish(nullptr)
			, _endOfStorage(nullptr)
		{
			reserve(n);
			while (n--)
			{
				push_back(value);
			}
		}
		/*
		* 理論上講,提供了vector(size_t n, const T& value = T())之后
		* vector(int n, const T& value = T())就不需要提供了,但是對(duì)于:
		* vector<int> v(10, 5);
		* 編譯器在編譯時(shí),認(rèn)為T(mén)已經(jīng)被實(shí)例化為int,而10和5編譯器會(huì)默認(rèn)其為int類(lèi)型
		* 就不會(huì)走vector(size_t n, const T& value = T())這個(gè)構(gòu)造方法,
		* 最終選擇的是:vector(InputIterator first, InputIterator last)
		* 因?yàn)榫幾g器覺(jué)得區(qū)間構(gòu)造兩個(gè)參數(shù)類(lèi)型一致,因此編譯器就會(huì)將InputIterator實(shí)例化為int
		* 但是10和5根本不是一個(gè)區(qū)間,編譯時(shí)就報(bào)錯(cuò)了
		* 故需要增加該構(gòu)造方法
		*/
		vector(int n, const T& value = T())
			: _start(new T[n])
			, _finish(_start + n)
			, _endOfStorage(_finish)
		{
			for (int i = 0; i < n; ++i)
			{
				_start[i] = value;
			}
		}
		// 若使用iterator做迭代器,會(huì)導(dǎo)致初始化的迭代器區(qū)間[first,last)只能是vector的迭代器
		// 重新聲明迭代器,迭代器區(qū)間[first,last)可以是任意容器的迭代器
		template<class InputIterator>
		vector(InputIterator first, InputIterator last)
		{
			while (first != last)
			{
				push_back(*first);
				++first;
			}
		}
		void swap(vector<T>& v)
		{
			std::swap(_start, v._start);
			std::swap(_finish, v._finish);
			std::swap(_endOfStorage, v._endOfStorage);
		}
		vector(const vector<T>& v)
			: _start(nullptr)
			, _finish(nullptr)
			, _endOfStorage(nullptr)
		{
			//現(xiàn)代寫(xiě)法,資本家寫(xiě)法
			vector<T> temp(v.begin(),v.end());
			swap(tmp);
		}
		vector<T>& operator=(vector<T> v)
		{
			swap(v);
			return *this;
		}
		~vector()
		{
			if (_start)
			{
				delete[] _start;
				_start = _finish = _endOfStorage = nullptr;
			}
		}
		/
		// 迭代器相關(guān)
		iterator begin()
		{
			return _start;
		}
		iterator end()
		{
			return _finish;
		}
		const_iterator cbegin() const
		{
			return _start;
		}
		const_iterator cend() const
		{
			return _finish;
		}
		//
		// 容量相關(guān)
		size_t size() const
		{
			return _finish - _start;
		}
		size_t capacity() const
		{
			return _endOfStorage - _start;
		}
		bool empty() const
		{
			return _start == _finish;
		}
		void reserve(size_t n)
		{
			if (n > capacity())
			{
				size_t oldSize = size();
				// 1. 開(kāi)辟新空間
				T* tmp = new T[n];
				// 2. 拷貝元素
				// 這里直接使用memcpy會(huì)有問(wèn)題嗎?請(qǐng)思考下
				//if (_start)
				//	memcpy(tmp, _start, sizeof(T)*size);
				if (_start)
				{
					for (size_t i = 0; i < oldSize; ++i)
						tmp[i] = _start[i];
					// 3. 釋放舊空間
					delete[] _start;
				}
				_start = tmp;
				_finish = _start + oldSize;
				_endOfStorage = _start + n;
			}
		}
		void resize(size_t n, const T& value = T())
		{
			// 1.如果n小于當(dāng)前的size,則數(shù)據(jù)個(gè)數(shù)縮小到n
			if (n <= size())
			{
				_finish = _start + n;
				return;
			}
			// 2.空間不夠則增容
			if (n > capacity())
				reserve(n);
			// 3.將size擴(kuò)大到n
			iterator it = _finish;
			_finish = _start + n;
			while (it != _finish)
			{
				*it = value;
				++it;
			}
		}
		///
		// 元素訪(fǎng)問(wèn)
		T& operator[](size_t pos)
		{
			assert(pos < size());
			return _start[pos];
		}
		const T& operator[](size_t pos)const
		{
			assert(pos < size());
			return _start[pos];
		}
		T& front()
		{
			return *_start;
		}
		const T& front()const
		{
			return *_start;
		}
		T& back()
		{
			return *(_finish - 1);
		}
		const T& back()const
		{
			return *(_finish - 1);
		}
		/
		// vector的修改操作
		iterator insert(iterator pos, const T& x)
		{
			assert(pos <= _finish);
			// 空間不夠先進(jìn)行增容
			if (_finish == _endOfStorage)
			{
				size_t n = pos - _start;
				size_t newCapacity = (0 == capacity()) ? 1 : capacity() * 2;
				reserve(newCapacity);
// 如果發(fā)生了增容,重新開(kāi)辟空間后,reserve會(huì)更新start和finish,但是不會(huì)更新pos,原空間被釋放掉,迭代器失效了,所以需要重置pos
				pos = _start + n;
			}
			iterator end = _finish - 1;
			while (end >= pos)
			{
				*(end + 1) = *end;
				--end;
			}
			*pos = x;
			++_finish;
			return pos;
		}
		// 返回刪除數(shù)據(jù)的下一個(gè)數(shù)據(jù)
		// 方便解決:一邊遍歷一邊刪除的迭代器失效問(wèn)題
		iterator erase(iterator pos)
		{
			// 挪動(dòng)數(shù)據(jù)進(jìn)行刪除
			iterator begin = pos + 1;
			while (begin != _finish) {
				*(begin - 1) = *begin;
				++begin;
			}
			--_finish;
			return pos;
		}
		void push_back(const T& x)//防止深拷貝,盡量用引用傳參
		{
			insert(end(), x);
		}
		void pop_back()
		{
			erase(end() - 1);
		}
	private:
		iterator _start;		// 指向數(shù)據(jù)塊的開(kāi)始
		iterator _finish;		// 指向最后有效數(shù)據(jù)的下一個(gè)位置
		iterator _endOfStorage;  // 指向存儲(chǔ)容量的尾
	};
}

使用memcpy拷貝問(wèn)題

假設(shè)模擬實(shí)現(xiàn)的vector中的reserve接口中,使用memcpy進(jìn)行的拷貝,以下代碼會(huì)發(fā)生什么問(wèn)題?

int main()
{
	bite::vector<swx::string> v;
	v.push_back("1111");
	v.push_back("2222");
	v.push_back("3333");
	return 0;
}

問(wèn)題分析:

  • memcpy是逐字節(jié)拷貝,將一段內(nèi)存空間中內(nèi)容原封不動(dòng)的拷貝到另外一段內(nèi)存空間中
  • 如果不涉及資源管理,memcpy既高效又不會(huì)出錯(cuò),但如果涉及到資源管理時(shí),就會(huì)出錯(cuò),因?yàn)閙emcpy的拷貝實(shí)際是淺拷貝。

如果對(duì)象中涉及到資源管理時(shí),千萬(wàn)不能使用memcpy進(jìn)行對(duì)象之間的拷貝,因?yàn)閙emcpy是淺拷貝,可能會(huì)引起一系列淺拷貝問(wèn)題,所以我們要使用賦值運(yùn)算符來(lái)完成,如果不涉及資源管理,那就正常賦值,如果涉及資源管理,那賦值運(yùn)算符中也已經(jīng)實(shí)現(xiàn)了深拷貝。

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

相關(guān)文章

  • Qt C++實(shí)現(xiàn)錄屏錄音功能的示例詳解

    Qt C++實(shí)現(xiàn)錄屏錄音功能的示例詳解

    實(shí)現(xiàn)一個(gè)錄屏+錄音的功能且需要快速開(kāi)發(fā),Qt無(wú)疑是一個(gè)非常好的選擇。他有豐富的類(lèi)庫(kù)和接口可以很好的滿(mǎn)足開(kāi)發(fā)需求。本文就來(lái)和大家聊聊具體的實(shí)現(xiàn)方法吧
    2023-03-03
  • c++ decltype關(guān)鍵字的用法

    c++ decltype關(guān)鍵字的用法

    這篇文章主要介紹了c++ decltype關(guān)鍵字的用法,幫助大家更好的理解和學(xué)習(xí)c++,感興趣的朋友可以了解下
    2020-10-10
  • C語(yǔ)言的隨機(jī)數(shù)rand()函數(shù)詳解

    C語(yǔ)言的隨機(jī)數(shù)rand()函數(shù)詳解

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言的隨機(jī)數(shù)rand()函數(shù),使用數(shù)據(jù)庫(kù),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助
    2022-02-02
  • C++模擬實(shí)現(xiàn)List迭代器詳解

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

    list不同于其他容器,他是一個(gè)鏈表,物理地址并不連續(xù)。所以在實(shí)現(xiàn)list類(lèi)的迭代器的時(shí)候,需要將迭代器單獨(dú)封裝到一個(gè)類(lèi)里,因?yàn)樾枰剌d很多操作符來(lái)跟其他容器的迭代器使用達(dá)成一致
    2022-04-04
  • 基于Windows API分解路徑問(wèn)題的詳解

    基于Windows API分解路徑問(wèn)題的詳解

    本篇文章是對(duì)Windows API分解路徑進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • C++中的類(lèi)模板詳解及示例

    C++中的類(lèi)模板詳解及示例

    我們?cè)诙x函數(shù)時(shí),可以通過(guò)定義函數(shù)模板,來(lái)簡(jiǎn)化一些功能相同而數(shù)據(jù)類(lèi)型不同的函數(shù)的定義和調(diào)用過(guò)程
    2013-10-10
  • C/C++ 凸多邊形求對(duì)角線(xiàn)交點(diǎn)的示例代碼

    C/C++ 凸多邊形求對(duì)角線(xiàn)交點(diǎn)的示例代碼

    這篇文章主要介紹了C/C++ 凸多邊形求對(duì)角線(xiàn)交點(diǎn)的示例代碼,代碼簡(jiǎn)單易懂,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-11-11
  • C語(yǔ)言中g(shù)etchar函數(shù)詳解看這一篇就夠了(函數(shù)功能、使用、返回值)

    C語(yǔ)言中g(shù)etchar函數(shù)詳解看這一篇就夠了(函數(shù)功能、使用、返回值)

    getchar讀取字符的函數(shù),今天通過(guò)本文給大家介紹C語(yǔ)言中g(shù)etchar函數(shù)簡(jiǎn)介用法示例詳解,感興趣的朋友跟隨小編一起看看吧
    2023-02-02
  • linux下基于C語(yǔ)言的信號(hào)編程實(shí)例

    linux下基于C語(yǔ)言的信號(hào)編程實(shí)例

    這篇文章主要介紹了linux下基于C語(yǔ)言的信號(hào)編程,實(shí)例分析了信號(hào)量的基本使用技巧與相關(guān)概念,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-07-07
  • C++常見(jiàn)異常處理原理及代碼示例解析

    C++常見(jiàn)異常處理原理及代碼示例解析

    這篇文章主要介紹了C++常見(jiàn)異常處理原理及代碼示例解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-07-07

最新評(píng)論