網(wǎng)站首頁 編程語言 正文
vector的模擬實現(xiàn)
#include <iostream>
using namespace std;
#include <assert.h>
namespace myVector
{
template<class T>
class vector
{
public:
// Vector的迭代器是一個原生指針
typedef T* iterator;
typedef const T* const_iterator;
///
// 構(gòu)造和銷毀
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())就不需要提供了,但是對于:
* vector<int> v(10, 5);
* 編譯器在編譯時,認(rèn)為T已經(jīng)被實例化為int,而10和5編譯器會默認(rèn)其為int類型
* 就不會走vector(size_t n, const T& value = T())這個構(gòu)造方法,
* 最終選擇的是:vector(InputIterator first, InputIterator last)
* 因為編譯器覺得區(qū)間構(gòu)造兩個參數(shù)類型一致,因此編譯器就會將InputIterator實例化為int
* 但是10和5根本不是一個區(qū)間,編譯時就報錯了
* 故需要增加該構(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做迭代器,會導(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)代寫法,資本家寫法
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. 開辟新空間
T* tmp = new T[n];
// 2. 拷貝元素
// 這里直接使用memcpy會有問題嗎?請思考下
//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ù)個數(shù)縮小到n
if (n <= size())
{
_finish = _start + n;
return;
}
// 2.空間不夠則增容
if (n > capacity())
reserve(n);
// 3.將size擴大到n
iterator it = _finish;
_finish = _start + n;
while (it != _finish)
{
*it = value;
++it;
}
}
///
// 元素訪問
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ā)生了增容,重新開辟空間后,reserve會更新start和finish,但是不會更新pos,原空間被釋放掉,迭代器失效了,所以需要重置pos
pos = _start + n;
}
iterator end = _finish - 1;
while (end >= pos)
{
*(end + 1) = *end;
--end;
}
*pos = x;
++_finish;
return pos;
}
// 返回刪除數(shù)據(jù)的下一個數(shù)據(jù)
// 方便解決:一邊遍歷一邊刪除的迭代器失效問題
iterator erase(iterator pos)
{
// 挪動數(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ù)塊的開始
iterator _finish; // 指向最后有效數(shù)據(jù)的下一個位置
iterator _endOfStorage; // 指向存儲容量的尾
};
}
使用memcpy拷貝問題
假設(shè)模擬實現(xiàn)的vector中的reserve接口中,使用memcpy進(jìn)行的拷貝,以下代碼會發(fā)生什么問題?
int main()
{
bite::vector<swx::string> v;
v.push_back("1111");
v.push_back("2222");
v.push_back("3333");
return 0;
}
問題分析:
- memcpy是逐字節(jié)拷貝,將一段內(nèi)存空間中內(nèi)容原封不動的拷貝到另外一段內(nèi)存空間中
- 如果不涉及資源管理,memcpy既高效又不會出錯,但如果涉及到資源管理時,就會出錯,因為memcpy的拷貝實際是淺拷貝。
如果對象中涉及到資源管理時,千萬不能使用memcpy進(jìn)行對象之間的拷貝,因為memcpy是淺拷貝,可能會引起一系列淺拷貝問題,所以我們要使用賦值運算符來完成,如果不涉及資源管理,那就正常賦值,如果涉及資源管理,那賦值運算符中也已經(jīng)實現(xiàn)了深拷貝。
原文鏈接:https://blog.csdn.net/weixin_44049823/article/details/128457721
- 上一篇:沒有了
- 下一篇:沒有了
相關(guān)推薦
- 2022-12-23 Kubernetes調(diào)度管理優(yōu)先級和搶占機制詳解_云其它
- 2022-11-06 react中braft-editor的基本使用方式_React
- 2022-07-12 如何利用python實現(xiàn)kmeans聚類_python
- 2022-06-01 C語言?深入淺出講解指針的使用_C 語言
- 2022-05-20 flume的負(fù)載均衡load balancer
- 2022-09-02 Qt為exe添加ico圖片的簡單實現(xiàn)步驟_C 語言
- 2022-04-29 DataTable的AcceptChanges()和RejectChanges()方法介紹并實現(xiàn)Da
- 2022-03-25 Unity實現(xiàn)圓形Image組件_C#教程
- 欄目分類
-
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支