当前位置: 首页 > news >正文

C++STL专题 vector底层实现


目录

一, vector的手搓

1.构造函数

2. 拷贝构造的实现

3.析构函数

4.begin() end() 的实现

5.reserve的实现

6.size和capacity的实现

7.push_back的实现

8.pop_back的实现

9.empty的实现

10.insert的实现

11.erase的实现

12.resize的实现

13.clear的实现

14.swap的实现

14. []重载

15.=重载

16. Vector类中的private

二,源码


一, vector的手搓

Vector类的定义

template <class T>
class Vector
{
public:typedef T* iterator;typedef const T* const_iterator;Vector(){}private:iterator _start = nullptr;//最开始iterator _finish = nullptr;//内容结尾(并不是容量的结尾)iterator _end_of_storage = nullptr;//容量结尾
};

1.构造函数

template<class InputIterator>
Vector(InputIterator first, InputIterator last)
{while (first != last){Push_back(*first);first++;}
}

(1).在模板类中再定义模板类,使得能够插入不同容器的值。

2. 拷贝构造的实现

Vector(const Vector<T>& v)//拷贝构造
{for (auto& e : v){Push_back(e);}
}

(1).直接使用范围for遍历一次即可。

3.析构函数

~Vector()
{if (_start){delete[] _start;_start = _finish = _end_of_storage = nullptr;}
}

(1).如果_start不空的话,就使用delete[] 释放空间。并且置空。

 

4.begin() end() 的实现

iterator begin()
{return _start;
}iterator end()
{return _finish;
}const_iterator begin() const
{return _start;
}const_iterator end() const
{return _finish;
}

(1).分为const修饰和不加const修饰,构成了函数重载,begin()返回起始_start即可,end()返回_finsih即可。

5.reserve的实现

void Reserve(size_t n)
{if (n > Capacity()){size_t oldsize = Size();T* tmp = new T[n];for (int i = 0; i < oldsize; i++) {tmp[i] = _start[i];}//memcpy(tmp, _start, Size() * sizeof(T));delete[] _start;_start = tmp;_finish = tmp + oldsize;_end_of_storage = tmp + n;}
}

(1).扩容函数,如果要求的空间大于当前的容量,就要扩容,首先存储当先数据个数Size()到oldsize,然后再开辟一个新空间,然后把_start中的全部数据拷贝到tmp中,然后再把_start释放空间,再将tmp赋值给_start,将_finish指向tmp+oldsize的位置。_end_of_storage指向tmp+n的位置。

6.size和capacity的实现

size_t Size()
{return _finish - _start;
}size_t Capacity()
{return _end_of_storage - _start;
}

(1).Size为vector中数据个数,所以直接返回_finish(最后一个数据的地址)减start(第一个数据的地址)即可。

(2).Capacity为vector中容量大小,所以直接返回_end_of_storage(容量的最后一个地址)减start(第一个数据的地址)即可。

7.push_back的实现

void Push_back(const T& x)
{if (_finish == _end_of_storage) // 扩容{Reserve(Capacity() == 0 ? 4 : Capacity() * 2);}*_finish = x;++_finish;
}

(1).push_back前要检查是否需要扩容,这里选择二倍扩容。

(2).直接将*_finish(元素的最后一个位置的下一个位置)赋值为x。再将_finish自加,使其移动到空位置,方便下次赋值。

8.pop_back的实现

void Pop_back()
{assert(!empty());--_finish;
}

(1).首先判断是否为空,如果为空就断言。

(2).直接将_finish自减即可。

9.empty的实现

bool empty()
{return _start == _finish;
}

(1).判断_start 和 _finish是否相等即可,如果相等就为空,否则为不空。

10.insert的实现

iterator insert(iterator pos, const T& x)
{assert(pos >= _start);assert(pos <= _finish);if (_finish == _end_of_storage) // 扩容{size_t len = pos - _start; // 解决迭代器失效Reserve(Capacity() == 0 ? 4 : Capacity() * 2);pos = _start + len; // 解决迭代器失效}iterator end = _finish - 1;while (end >= pos) // 迭代器失效问题已经扩容了,但是pos还是指向原来的旧空间 野指针{*(end + 1) = *end;end--;}*pos = x;++_finish;return pos;
}

(1).大致思路为把要插入位置(pos)之后的全部向后移动一位,然后再将pos位置赋值为要插入的元素。

(2).要想插入,首先判断是否需要扩容。

(3).定义一个迭代器end,存储vector中最后一个元素的位置,然后循环判断(end >= pos),每次成立时,就把end+1处赋值为end的元素,然后将end自减。退出循环后,直接将*pos赋值为x并把_finish自加即可。

11.erase的实现

void erase(iterator pos)
{assert(pos >= _start);assert(pos <= _finish);iterator it = pos + 1;while (it != end()){*(it - 1) = *it;++it;}--_finish;
}

(1).大致思路为将pos位置之后的全部元素都向前移动一个位置,最后将_finish自减即可。

12.resize的实现

void resize(size_t n, T val = T())
{// n<size// size<n<capacity// n>capaciityif (n < Size()){_finish = _start + n;}else{Reserve(n);while (_finish < _start + n){*_finish = val;++_finish;}}
}

(1).resize为对字符串大小做改变,当n<size时,就会对当前字符串缩减,其余情况为reserve。

13.clear的实现

void clear()
{_finish = _start;
}

(1).直接将_finish=_start即可。

14.swap的实现

void swap(Vector<T>& v)
{std::swap(_start, v._start);std::swap(_finish, v._finish);std::swap(_end_of_storage, v._end_of_storage);
}

(1).直接交换即可。

14. []重载

T& operator[](size_t i)
{return _start[i];
}const T& operator[](size_t i) const
{return _start[i];
}

(1).由于_start为顺序表,所以直接返回顺序表的值即可。

15.=重载

//方法一:Vector<T>& operator=(const Vector<T>& v)
{if (this != &v){clear();Reverse(v.Size());for (auto& e : v){Push_back(e);}}return *this;
}//方法二:Vector<T>& operator=( vector<T> v){swap(v);return *this;}

(1).有两种方法,第一种方法为先清空_start,然后再对_start重新扩容,扩容置与v相等,然后遍历v,把v中的元素按个插入_start。

(2).方法二:要进行传值传参,不能用传址传参,然后直接交换即可,这样_start就成了v了,由于v为传值传参,并不受影响。

16. Vector类中的private

private:iterator _start = nullptr;//缺省iterator _finish = nullptr;//缺省iterator _end_of_storage = nullptr;//缺省

二,源码

#pragma once
#include <iostream>
#include <vector>
#include <assert.h>
#include<algorithm>namespace hhc
{template <class T>class Vector{public:typedef T* iterator;typedef const T* const_iterator;Vector(){}Vector(const Vector<T>& v)//拷贝构造{for (auto& e : v){Push_back(e);}}~Vector(){if (_start){delete[] _start;_start = _finish = _end_of_storage = nullptr;}}iterator begin(){return _start;}iterator end(){return _finish;}const_iterator begin() const{return _start;}const_iterator end() const{return _finish;}void Reserve(size_t n){if (n > Capacity()){size_t oldsize = Size();T* tmp = new T[n];for (int i = 0; i < oldsize; i++) {tmp[i] = _start[i];}//memcpy(tmp, _start, Size() * sizeof(T));delete[] _start;_start = tmp;_finish = tmp + oldsize;_end_of_storage = tmp + n;}}size_t Size(){return _finish - _start;}size_t Capacity(){return _end_of_storage - _start;}void Push_back(const T& x){if (_finish == _end_of_storage) // 扩容{Reserve(Capacity() == 0 ? 4 : Capacity() * 2);}*_finish = x;++_finish;}bool empty(){return _start == _finish;}void Pop_back(){assert(!empty());--_finish;}void insert(iterator pos, const T& x){assert(pos >= _start);assert(pos <= _finish);if (_finish == _end_of_storage) // 扩容{size_t len = pos - _start; // 解决迭代器失效Reserve(Capacity() == 0 ? 4 : Capacity() * 2);pos = _start + len; // 解决迭代器失效}iterator end = _finish - 1;while (end >= pos) // 迭代器失效问题已经扩容了,但是pos还是指向原来的旧空间 野指针{*(end + 1) = *end;end--;}*pos = x;++_finish;}void erase(iterator pos){assert(pos >= _start);assert(pos <= _finish);iterator it = pos + 1;while (it != end()){*(it - 1) = *it;++it;}--_finish;}void resize(size_t n, T val = T()){// n<size// size<n<capacity// n>capaciityif (n < Size()){_finish = _start + n;}else{Reserve(n);while (_finish < _start + n){*_finish = val;++_finish;}}}void clear(){_finish = _start;}void swap(Vector<T>& v){std::swap(_start, v._start);std::swap(_finish, v._finish);std::swap(_end_of_storage, v._end_of_storage);}template<class InputIterator>Vector(InputIterator first, InputIterator last){while (first != last){Push_back(*first);first++;}}T& operator[](size_t i){return _start[i];}const T& operator[](size_t i) const{return _start[i];}//方法一:Vector<T>& operator=(const Vector<T>& v){if (this != &v){clear();Reverse(v.Size());for (auto& e : v){Push_back(e);}}return *this;}//方法二:Vector<T>& operator=( vector<T> v){swap(v);return *this;}private:iterator _start = nullptr;iterator _finish = nullptr;iterator _end_of_storage = nullptr;};template <class Container>void print_Container(const Container& v){// 没用实例化的类模板里面取东西,编译器不能区分const_iterator是类型还是静态成员变量// 编译器规定不能去没用实例化的模板里面取东西typename Container::const_iterator it = v.begin();auto itt = v.begin(); // 这样也行while (it != v.end()){std::cout << *it << ' ';*it++;}std::cout << std::endl;for (auto num : v){std::cout << num << ' ';}}}

本篇完

相关文章:

C++STL专题 vector底层实现

目录 一&#xff0c; vector的手搓 1.构造函数 2. 拷贝构造的实现 3.析构函数 4.begin() end() 的实现 5.reserve的实现 6.size和capacity的实现 7.push_back的实现 8.pop_back的实现 9.empty的实现 10.insert的实现 11.erase的实现 12.resize的实现 13.clear的实…...

【Linux】装机常用配置

文章目录 1. 下载常用软件包2. 更新yum源3. vim编辑器配置4. 安装C语言和C的静态库&#xff08;换root&#xff09;5. git6. sudo给普通用户提权7. 更新git版本&#xff08;centos默认安装1.8.x&#xff0c;我们更新到2.x&#xff09;8. getch9. json10. 升级gcc版本11. 跨系统…...

oracle库PASSWORD_VERSIONS 对应的加密方式

oracle库PASSWORD_VERSIONS 对应的加密方式 10G DES 11G SHA-1 12C SHA-2-based SHA-512官方文档&#xff1a; https://docs.oracle.com/database/121/DBSEG/authentication.htm#DBSEG487...

分享一个基于微信小程序的乡村医疗上门服务预约平台(源码、调试、LW、开题、PPT)

&#x1f495;&#x1f495;作者&#xff1a;计算机源码社 &#x1f495;&#x1f495;个人简介&#xff1a;本人 八年开发经验&#xff0c;擅长Java、Python、PHP、.NET、Node.js、Android、微信小程序、爬虫、大数据、机器学习等&#xff0c;大家有这一块的问题可以一起交流&…...

切香肠(Sausage)

题目描述 有 n 条香肠&#xff0c;每条香肠的长度相等。我们打算将这些香肠切开后分给 k 名客人&#xff0c;且要求每名客人获得一样多的香肠&#xff0c;且要将所有的香肠分配完&#xff0c;不做保留。 请问最少需要切几刀才能完成&#xff1f;一刀只能切断一条香肠&#xf…...

Session与Cookie以及Cache区别,及应用场景

Session、Cookie和Cache是Web开发中常用的数据存储方式&#xff0c;它们在功能、存储位置和应用场景上有所不同。 一、Session、Cookie和Cache的区别 Session 存储位置&#xff1a;服务器端。功能&#xff1a;通过在服务器上存储唯一的标识符&#xff08;Session ID&#xff…...

Debian | 更换 Gnome 至 Xfce4

Debian | 更换 Gnome 至 Xfce4 更新源 sudo apt update && sudo apt upgrade安装 xfce4 sudo apt install xfce4我选择 lightdm&#xff0c;回车 切换桌面 sudo update-alternatives --config x-session-manager输入 xfce 所在序号&#xff0c;我这里是 3 卸载 …...

在使用JSON过程中遇到的一个空间释放问题

在对完成的模块进行空间访问检查中发现了这个问题&#xff0c;这刚开始接触JSON的使用&#xff0c;也不知道他的内部实现&#xff0c;因此该问题找了好久&#xff0c;终于发现是每个节点创建都会自动开辟空间&#xff0c;因此造成空间未成功释放的错误。 JSON未成功替换节点空间…...

基于ThinkPHP开发的校园跑腿社区小程序系统源码,包含前后端代码

基于ThinkPHP开发的校园跑腿社区小程序系统源码&#xff0c;包含前后端代码 最新独立版校园跑腿校园社区小程序系统源码 | 附教程 测试环境&#xff1a;NginxPHP7.2MySQL5.6 多校版本&#xff0c;多模块&#xff0c;适合跑腿&#xff0c;外卖&#xff0c;表白&#xff0c;二…...

不同专业方向如何在ChatGPT的帮助下完成选题

学境思源&#xff0c;一键生成论文初稿&#xff1a; AcademicIdeas - 学境思源AI论文写作 选择一个合适的论文题目是每个论文写作同学必须面对的重要任务。无论是历史专业、计算机科学专业&#xff0c;还是其他各个领域&#xff0c;找到一个既有研究价值又符合个人兴趣的选题往…...

MathType7.4中文版本功能详解!你的数学公式编辑神器

嘿&#xff0c;亲爱的小伙伴们&#xff0c;今天我要跟大家分享一个超实用的工具——MathType7中文版。作为一个自媒体人&#xff0c;我常常需要编辑各种复杂的数学公式&#xff0c;而这款软件简直就是我的救星&#xff01;接下来&#xff0c;就让我带你们领略一下它的神奇之处吧…...

在 PhpStorm 中为 .java 文件启用语法高亮,需要正确配置文件类型和关联语言。

点击访问我的技术博客https://ai.weoknow.comhttps://ai.weoknow.com 因为我同时使用java和php混编所以在一个项目中如果同时打开IntelliJ IDEA和PhpStorm不符合我完美主义的本性。 捣鼓了一下搞定了 1. 添加文件类型关联 将 .java 文件与 Java 语言支持关联&#xff1a; …...

2024年8月1日(前端服务器的配置以及tomcat环境的配置)

[rootstatic ~]# cd eleme_web/ [rootstatic eleme_web]# cd src/ [rootstatic src]# ls views/ AboutView.vue HomeView.vue [rootstatic src]# vim views/HomeView.vue [rootstatic src]# nohup npm run serve nohup: 忽略输入并把输出追加到"nohup.out" 构建项目…...

基于tcp,html,数据库的在线信息查询系统项目总结

1.项目背景 在线信息查询系统是一种可用于检索和展示各种信息的计算机程序或平台。主要特点包括&#xff1a; 用户接口&#xff1a;通常提供友好的界面&#xff0c;用户可以方便地输入查询条件。 数据存储&#xff1a;系统往往连接到数据库&#xff0c;存储大量信息&#xf…...

P1032 [NOIP2002 提高组] 字串变换

[NOIP2002 提高组] 字串变换 题目背景 本题不保证存在靠谱的多项式复杂度的做法。测试数据非常的水&#xff0c;各种做法都可以通过&#xff0c;不代表算法正确。因此本题题目和数据仅供参考。 本题为搜索题&#xff0c;本题不接受 hack 数据。关于此类题目的详细内容 题目…...

Android 12系统源码_多屏幕(一)多屏幕设备显示Activity

前言 分屏&#xff1a;是指一个屏幕分出多个窗口&#xff0c;分别显示不同应用的界面&#xff0c;这在当前的手机设备中很常见。多屏&#xff1a;是指一个设备存在多个屏幕&#xff0c;这些可能是虚拟屏幕或者实体硬件屏幕&#xff0c;不同的应用同时显示在不同的屏幕中&#…...

如何判断IP地址属于住宅IP还是机房IP

在数字化时代,IP地址作为互联网通信的基础标识&#xff0c;扮演着重要的角色。无论是网络管理、数据分析还是安全监控&#xff0c;正确识别IP地址的类型——尤其是区分是住宅IP还是机房IP&#xff0c;对于确保网络安全、优化网络性能以及合法合规运营具有重要意义。IPIDEA代理I…...

C#TreeView控件应用

1、代码 using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.IO; using System.Linq; using System.Text; using System.Threading.Tasks; using System.Windows.Forms;namespace TestApp…...

计算机网络-数据链路层

基本概念 数据链路和链路 链路&#xff1a;指的是从一个节点到相邻节点的一段物理线路&#xff0c;且中间没有任何其他的交换节点 数据链路&#xff1a;传输数据时&#xff0c;除了一条物理线路&#xff0c;还需要一些必要通信协议来控制这些传输。 数据链路层的三个基本问…...

农场游戏中的时间管理实例

一、准备工作 在Unity中创建承载日期和时间的文本 二、设置游戏的时间戳 using System.Collections; using System.Collections.Generic; using UnityEngine; //标识这个类可以被序列化 [System.Serializable] public class GameTimestamp {// 游戏时间戳的成员变量public in…...

AI-调查研究-01-正念冥想有用吗?对健康的影响及科学指南

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; &#x1f680; AI篇持续更新中&#xff01;&#xff08;长期更新&#xff09; 目前2025年06月05日更新到&#xff1a; AI炼丹日志-28 - Aud…...

HTML 语义化

目录 HTML 语义化HTML5 新特性HTML 语义化的好处语义化标签的使用场景最佳实践 HTML 语义化 HTML5 新特性 标准答案&#xff1a; 语义化标签&#xff1a; <header>&#xff1a;页头<nav>&#xff1a;导航<main>&#xff1a;主要内容<article>&#x…...

智慧工地云平台源码,基于微服务架构+Java+Spring Cloud +UniApp +MySql

智慧工地管理云平台系统&#xff0c;智慧工地全套源码&#xff0c;java版智慧工地源码&#xff0c;支持PC端、大屏端、移动端。 智慧工地聚焦建筑行业的市场需求&#xff0c;提供“平台网络终端”的整体解决方案&#xff0c;提供劳务管理、视频管理、智能监测、绿色施工、安全管…...

大数据零基础学习day1之环境准备和大数据初步理解

学习大数据会使用到多台Linux服务器。 一、环境准备 1、VMware 基于VMware构建Linux虚拟机 是大数据从业者或者IT从业者的必备技能之一也是成本低廉的方案 所以VMware虚拟机方案是必须要学习的。 &#xff08;1&#xff09;设置网关 打开VMware虚拟机&#xff0c;点击编辑…...

React19源码系列之 事件插件系统

事件类别 事件类型 定义 文档 Event Event 接口表示在 EventTarget 上出现的事件。 Event - Web API | MDN UIEvent UIEvent 接口表示简单的用户界面事件。 UIEvent - Web API | MDN KeyboardEvent KeyboardEvent 对象描述了用户与键盘的交互。 KeyboardEvent - Web…...

Android15默认授权浮窗权限

我们经常有那种需求&#xff0c;客户需要定制的apk集成在ROM中&#xff0c;并且默认授予其【显示在其他应用的上层】权限&#xff0c;也就是我们常说的浮窗权限&#xff0c;那么我们就可以通过以下方法在wms、ams等系统服务的systemReady()方法中调用即可实现预置应用默认授权浮…...

R语言速释制剂QBD解决方案之三

本文是《Quality by Design for ANDAs: An Example for Immediate-Release Dosage Forms》第一个处方的R语言解决方案。 第一个处方研究评估原料药粒径分布、MCC/Lactose比例、崩解剂用量对制剂CQAs的影响。 第二处方研究用于理解颗粒外加硬脂酸镁和滑石粉对片剂质量和可生产…...

Python基于历史模拟方法实现投资组合风险管理的VaR与ES模型项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档&#xff09;&#xff0c;如需数据代码文档可以直接到文章最后关注获取。 1.项目背景 在金融市场日益复杂和波动加剧的背景下&#xff0c;风险管理成为金融机构和个人投资者关注的核心议题之一。VaR&…...

现有的 Redis 分布式锁库(如 Redisson)提供了哪些便利?

现有的 Redis 分布式锁库&#xff08;如 Redisson&#xff09;相比于开发者自己基于 Redis 命令&#xff08;如 SETNX, EXPIRE, DEL&#xff09;手动实现分布式锁&#xff0c;提供了巨大的便利性和健壮性。主要体现在以下几个方面&#xff1a; 原子性保证 (Atomicity)&#xff…...

C++.OpenGL (20/64)混合(Blending)

混合(Blending) 透明效果核心原理 #mermaid-svg-SWG0UzVfJms7Sm3e {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-SWG0UzVfJms7Sm3e .error-icon{fill:#552222;}#mermaid-svg-SWG0UzVfJms7Sm3e .error-text{fill…...