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

【C++】vector模拟实现

在这里插入图片描述

🔥个人主页: Forcible Bug Maker
🔥专栏: STL || C++

目录

  • 前言
  • 🔥vector需要实现的接口函数
  • 🔥vector的模拟实现
    • ==swap交换==
    • ==默认成员函数==
    • ==迭代器接口==
    • ==reserve和resize==
    • ==size和capacity==
    • ==operator[ ]下标获取==
    • ==push_back和pop_back==
    • ==插入(insert)和删除(erase)==
  • 结语

前言

本篇博客主要内容:STL库中vector的模拟实现

在之前完成string以及学习了vector一些接口函数的基础上,这个vector的实现相当于是一个奖励内容,并不困难。不过我们这里vector底层实现和上次string的有所不同,是通过三个指针_start_finish_end_of_storage来维护这个模板类的。相信在看完今天vector的实现之后,能对C++的迭代器有更深的了解。

🔥vector需要实现的接口函数

由于涉及到了模板的内容,我们这次不会将vector实现的声明和定义分离。在后期模板进阶的阶段会进一步解答此问题,盲目将模板类的声明定义分离是很容易出错的(在对模板这部分内容不熟练的情况下)。
看看需要实现的接口函数:

#pragma once
#include<iostream>
#include<algorithm>
#include<cassert>
using namespace std;
namespace ForcibleBugMaker
{template<class T>class vector{public:// 这里vector的迭代器是一个原生指针typedef T* iterator;typedef const T* const_iterator;//迭代器获取接口iterator begin();iterator end()const_iterator cbegin()const;const_iterator cend() const;// 交换vector对象void swap(vector<T>& v);// 构造函数,析构函数以及赋值运算符重载vector();vector(int n, const T& value = T());template<class InputIterator>vector(InputIterator first, InputIterator last);vector(const vector<T>& v);vector<T>& operator=(vector<T> v);~vector();// 容量获取和调整接口size_t size() const;size_t capacity() const;void reserve(size_t n);void resize(size_t n, const T& value = T());// 元素获取T& operator[](size_t pos);const T& operator[](size_t pos)const;// 插入和删除元素void push_back(const T& x);void pop_back();iterator insert(iterator pos, const T& x);iterator erase(iterator pos);private:iterator _start = nullptr; // 指向数据块的开始iterator _finish = nullptr; // 指向有效数据的尾iterator _end_of_storage = nullptr; // 指向存储容量的尾};
}

本篇建议在string类实现的基础上进行,不然有些概念可能不太好理解。C++中包含交换函数swap,在命名空间std中,可以直接使用。在vector的实现中,你可能发现与string不同,接口函数大多使用迭代器(指针)来实现,传入和传出的参数基本上也都是迭代器。这么做是为了给后面一系列的STL容器打一个基础,大家要逐渐习惯这样的实现方式。

🔥vector的模拟实现

接下来进入主要内容,按照上面的接口列表逐一实现。

本篇都是直接在vector模板类内部实现,不用手动圈定命名空间。

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);
}

默认成员函数

构造函数:
这里我们提供了三种重载,分别是:

// 无参构造
vector()
{}// 构造包含n个value的vector对象
vector(int n, const T& value = T())
{reserve(n);for (int i = 0; i < n; i++) {this->push_back(value);}
}// 通过迭代器区间构造vector对象
template<class InputIterator>
vector(InputIterator first, InputIterator last)
{InputIterator it = first;while (it != last) {this->push_back(*it);++it;}
}

学习过vector的使用,应该也能看懂。其中有几个还待实现的接口函数,如push_backreserve等。

对于第一个无参构造,其实编译器默认实现的也有相同的效果,但是一旦你实现了下面两个构造,那么编译器就不会生成默认的了。C++提供了一种语法,可以无视你的重载构造,强制生成一个编译器默认构造,所以,下面这样写也是正确的的。

 // 强制编译器生成默认无参构造vector() = default;vector(int n, const T& value = T()){reserve(n);for (int i = 0; i < n; i++) {this->push_back(value);}}template<class InputIterator>vector(InputIterator first, InputIterator last){InputIterator it = first;while (it != last) {this->push_back(*it);++it;}}

第二个构造中缺省参数T()表示的是构造一个T类型的对象,赋值给value。在C++中不但T()可以用于表示自定义类型变量的无参构造,也同样可以表示内置类型的无参构造(C语言中并不支持这样做)。
如下:

int a = int();
int b = int(3);
cout << a << endl;
cout << b << endl;

在这里插入图片描述

对于第三个,迭代器构造,实现的是一个模板成员函数,支持各种类型迭代器的区间构造。

析构函数:

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

拷贝构造:

vector(const vector<T>& v)
{reserve(v.capacity());int size = v.size();for (int i = 0; i < size; i++) {this->push_back(v[i]);}
}

赋值运算符重载:

vector<T>& operator=(vector<T> v)
{swap(v);return *this;
}

这是一种比较新的实现方式,在string中也有使用过。

迭代器接口

本篇vector虽然使用指针实现了迭代器,但并不是说只有这一种实现方式。你也可以将迭代器定义成一个模板类,具体内容可以在不久后的list篇章学到。

typedef T* iterator;
typedef const T* const_iterator;iterator begin()
{return _start;
}iterator end()
{return _finish;
}const_iterator cbegin()const
{return _start;
}const_iterator cend() const
{return _finish;
}

reserve和resize

reserve开辟容量空间capacity,但不改变vector对象原本的size区间中的内容。

void reserve(size_t n)
{if (n > capacity()) {T* tmp = new T[n];size_t pos = size();for (int i = 0; i < size(); i++) {tmp[i] = _start[i];}delete[] _start;_start = tmp;_finish = _start + pos;_end_of_storage = _start + n;}
}

resize也可以用作开辟空间,同时会改变size的值。

void resize(size_t n, const T& value = T())
{if (n > capacity()) {reserve(n);}while (_finish < _start + n) {*_finish = value;++_finish;}_finish = _start + n;
}

size和capacity

获取vector对象的size和capacity,通过指针相减就可以得到。

size_t size() const
{return _finish - _start;
}size_t capacity() const
{return _end_of_storage - _start;
}

operator[ ]下标获取

通过运算符重载的operator[]获取vector对象的元素。

T& operator[](size_t pos)
{assert(pos < size());return _start[pos];
}const T& operator[](size_t pos)const
{assert(pos < size());return _start[pos];
}

同时重载了非const版本和const版本。

push_back和pop_back

push_backpop_back分别是尾插一个元素和尾删一个元素。
复用之前的reserve接口函数push_back功能其实并不难实现。

void push_back(const T& x)
{if (_finish == _end_of_storage) {reserve(size() == 0 ? 4 : size() * 2);}*_finish = x;++_finish;
}void pop_back()
{assert(size() > 0);--_finish;
}

插入(insert)和删除(erase)

这两个函数需要传入的参数都为迭代器,从vector中间部分插入和删除元素。

插入:会将我们传入的元素插入到迭代器指向的对象之前;如果迭代器指向end(),则功能等同于尾插。
删除:将传入迭代器指向的元素删除。

iterator insert(iterator pos, const T& x)
{assert(_start <= pos && pos <= _finish);if (_finish == _end_of_storage) {int gap = pos - _start;reserve(size() == 0 ? 4 : size() * 2);pos = _start + gap;}iterator it = _finish;while (it > pos) {*it = *(it - 1);--it;}*it = x;++_finish;return it + 1;
}iterator erase(iterator pos)
{assert(_start <= pos && pos < _finish);int size = pos - _start;iterator itCur = pos + 1;while (itCur != _finish) {*(itCur - 1) = *itCur;++itCur;}--_finish;return _start + size;
}

注:vector的insert和erase可能导致大量数据的移动,开销较大,所以非必要情况少用。

结语

本篇博客主要介绍了vector常用接口的实现,包括迭代器以及迭代器接口,元素的插入和删除,以及各种默认成员函数的实现。
后续博主还会继续分享与STL相关的内容,感谢大家的支持。♥

相关文章:

【C++】vector模拟实现

&#x1f525;个人主页&#xff1a; Forcible Bug Maker &#x1f525;专栏&#xff1a; STL || C 目录 前言&#x1f525;vector需要实现的接口函数&#x1f525;vector的模拟实现swap交换默认成员函数迭代器接口reserve和resizesize和capacityoperator[ ]下标获取push_back和…...

生成随机图片

package com.zhuguohui.app.lib.tools;/*** Created by zhuguohui* Date: 2024/6/1* Time: 13:39* Desc:获取随机图片*/ public class RandomImage {// static final String url "https://picsum.photos/%d/%d?random%d";static final String url "https://…...

回溯算法常见思路

回溯问题 回溯法&#xff0c;一般可以解决如下几种问题&#xff1a; 组合问题&#xff1a;N个数里面按一定规则找出k个数的集合切割问题&#xff1a;一个字符串按一定规则有几种切割方式子集问题&#xff1a;一个N个数的集合里有多少符合条件的子集排列问题&#xff1a;N个数…...

AR眼镜定制开发_在AR眼镜中实现ChatGPT功能

AR眼镜定制方案中&#xff0c;需要考虑到强大的算力、轻巧的设计和更长的续航时间等基本要求。然而&#xff0c;AR眼镜的设计方案不仅仅需要在硬件和显示技术方面取得突破&#xff0c;还要在用户体验方面有所进展。 过去&#xff0c;由于造价较高&#xff0c;AR眼镜的普及和商业…...

手写防抖debounce

手写防抖debounce 应用场景 当需要在事件频繁触发时&#xff0c;只执行最后一次操作&#xff0c;可以使用防抖函数来控制函数的执行频率,比如窗口resize事件和输入框input事件&#xff1b; 这段代码定义了一个名为 debounce 的函数&#xff0c;它接收两个参数&#xff1a;fn…...

anaconda pycharm jupter分别是

Anaconda Anaconda是一个面向数据科学的Python发行版&#xff0c;它包含了Python解释器、conda包管理器、以及大量的科学计算和数据分析库。Anaconda的主要功能是提供一个易于管理的环境&#xff0c;用于安装、运行和更新Python包&#xff0c;同时支持创建和切换不同的Python环…...

【JMeter接口自动化】第3讲 Jmeter语言及外观配置

Jmeter语言配置 方法一&#xff1a;暂时生效&#xff0c;下次打开JMeter还会恢复默认配置 Jmeter安装后&#xff0c;默认语言是英文&#xff0c;可以在“选项”——“选择语音”中更改 方法二&#xff0c;修改配置文件&#xff0c;永久生效 修改jmeter.properties文件 Jmete…...

浅谈云原生安全

一、云原生安全的层级概念 "4C" Code-Container-Cluster-Cloud 二、云原生各个层级的安全实践有哪些&#xff1f; 1、针对于Cloud针对的是公有云层面&#xff0c;其实就一点 1、使用主账号子角色&#xff0c;赋予最小权限原则进行资源管理。 2、对于Cluster 1、从C…...

[线程与网络] 网络编程与通信原理(五): 深入理解网络层IP协议与数据链路层以太网协议

&#x1f338;个人主页:https://blog.csdn.net/2301_80050796?spm1000.2115.3001.5343 &#x1f3f5;️热门专栏:&#x1f355; Collection与数据结构 (92平均质量分)https://blog.csdn.net/2301_80050796/category_12621348.html?spm1001.2014.3001.5482 &#x1f9c0;Java …...

【Python】超时请求或计算的处理

超时机制 一般应用于处理阻塞问题 场景&#xff1a; 复杂度较大的计算&#xff08;解析&#xff09;某个数值、加解密计算等请求中遇到阻塞&#xff0c;避免长时间等待网络波动&#xff0c;避免长时间请求&#xff0c;浪费时间 1. requests 请求超时机制 reqeusts 依赖中的…...

前端 JS 经典:LRU 缓存算法

前言&#xff1a;什么是 LRU 呢&#xff0c;单词全拼 Least Recently Used&#xff0c;意思是最久未使用。这个算法是做缓存用的&#xff0c;比如&#xff0c;你要缓存一组数据&#xff0c;你要划分缓存块出来&#xff0c;因为不可能每个数据都做缓存&#xff0c;那么划出来的这…...

Python3 笔记:ljust、rjust 和 center

1、ljust() 方法返回一个原字符串左对齐,并使用空格填充至指定长度的新字符串。如果指定的长度小于原字符串的长度则返回原字符串。 ljust(self, width, fillchar , /) width -- 指定字符串长度。 fillchar -- 填充字符&#xff0c;默认为空格。 str1 sentence str2 word…...

恒压频比开环控制系统Matlab/Simulink仿真分析(SPWM控制方式)

介绍恒压频比的开环控制方法驱动永磁同步电机的转动&#xff0c;首先分析恒压频比的控制原理&#xff0c;然后在Matlab/Simulink中进行永磁同步电机恒压频比开环控制系统的仿真分析&#xff0c;最后将Simulink中的恒压频比控制算法生成代码加载到实际工程中进行工程实现。 一、…...

chatgpt4和文心一言的简单对比

功能1. 写一个ebpf程序监控udp延迟 文心一言 答案 编写一个完整的eBPF程序来监控UDP延迟涉及多个步骤&#xff0c;包括编写eBPF C代码、编译、加载到内核&#xff0c;以及用户空间程序来读取eBPF map中的数据。以下是一个简化的示例框架&#xff0c;展示如何使用eBPF来监控U…...

React 为什么使用map来渲染列表 而不是其他循环方法

1. 声明式与函数式编程 React强调声明式编程&#xff0c;这意味着你只需要关心代码“做什么”&#xff0c;而不是“怎么做”。.map()函数是一种高阶函数&#xff0c;它属于函数式编程范畴&#xff0c;能够返回一个新数组&#xff0c;这非常适合用于生成组件列表。 使用.map()…...

【Axure高保真】tab切换输入表单

今天和大家分享tab切换输入表单的原型模板&#xff0c;这个模板方便我们快速制作表单&#xff0c;里面包含了输入框、下拉列表、选择器共10多种常用的元件&#xff0c;后续也可以根据需要自行添加到中继器里。点击tab标签可以分类填写对应的内容&#xff0c;这个原型模板是用中…...

OrangePi AI Pro 测试体验

感谢CSDN活动提供的OrangePi AI Pro &#xff0c;之前一直用的树莓派&#xff0c;正好体验一下新的国产设备&#xff0c; 1、开机体验 整个设备包装不错&#xff0c;链接键盘、屏幕和鼠标&#xff0c;整体开机体验不错&#xff0c;内置OS不错&#xff0c;这个系统内嵌了中文输…...

【C++】:模板初阶和STL简介

目录 一&#xff0c;泛型编程二&#xff0c;函数模板2.1 函数模板概念2.2 函数模板格式2.3 函数模板的原理2.4 函数模板的实例化2.5 模板参数的匹配原则 三&#xff0c;类模板3.1 类模板的定义格式3.2 类模板的实例化 四&#xff0c;STL简介&#xff08;了解&#xff09;4.1 什…...

【软件开发】Java学习路线

本路径视频教程均来自尚硅谷B站视频&#xff0c;Java学习课程我已经收藏在一个文件夹下&#xff0c;B站文件夹同时会收藏其他Java视频&#xff0c;感谢关注。指路&#xff1a;https://www.bilibili.com/medialist/detail/ml3113981545 2024Java学习路线&#xff08;快速版&…...

git拉去代码报错“Failed to connect to 127.0.0.1 port 31181: Connection refused“

最近参与了一个新项目&#xff0c;在使用git clone 克隆代码时遇到了一个报错"fatal: unable to access ‘https://example.git/’: Failed to connect to 127.0.0.1 port 31181: Connection refused",今天就和大家分享下解决过程。 报错详情 在使用git clone 克隆…...

解读信创产业根基,操作系统发展历程

信创产业根基之一操作系统 操作系统是一个关键的控制程序&#xff0c;负责协调、管理和控制计算机硬件和软件资源。作为硬件的首要软件扩展&#xff0c;它位于裸机与用户之间&#xff0c;充当了两者之间的桥梁。通过其核心程序&#xff0c;操作系统高效地管理着系统中的各类资源…...

使用Python爬取华为市场游戏类APP应用

文章目录 1. 写在前面2. 接口分析3. 爬虫开发4. 下载链接获取 【&#x1f3e0;作者主页】&#xff1a;吴秋霖 【&#x1f4bc;作者介绍】&#xff1a;擅长爬虫与JS加密逆向分析&#xff01;Python领域优质创作者、CSDN博客专家、阿里云博客专家、华为云享专家。一路走来长期坚守…...

【Oracle】修改已经存在的序列的当前值

前情提要 在oracle中一般使用序列来实现ID自增。但是oracle中序列维护的没有mysql那么好。只是单存的递增。 比如新建了一个序列&#xff0c;从1开始&#xff0c;每次递增1。此时我向数据库里插入一条id10的数据。那么在序列查询到10的时候&#xff0c;插入就会报错。 所以比较…...

记一次netty客户端的开发

背景 近日要开发一个tcp客户端程序去对接上游厂商的数据源&#xff0c;决定使用netty去处理&#xff0c;由于很久没有开发过netty了&#xff0c;顺便学习记录下 netty搭建 考虑到我们需要多个client去对接server服务&#xff0c;所以我们定义一个公共的AbstractNettyClient父…...

策略模式结合Spring使用

1.抽象策略 /*** 支付方式策略* author Linging* version 1.0.0* since 1.0*/ public interface PayStrategy {void pay(BigDecimal money);}2.具体策略 /*** 支付宝* author Linging* version 1.0.0* since 1.0*/ Component("aliPayStrategy") public class AliPa…...

基于 RNNs 对 IMDB 电影评论进行情感分类

前言 系列专栏:【深度学习:算法项目实战】✨︎ 涉及医疗健康、财经金融、商业零售、食品饮料、运动健身、交通运输、环境科学、社交媒体以及文本和图像处理等诸多领域,讨论了各种复杂的深度神经网络思想,如卷积神经网络、循环神经网络、生成对抗网络、门控循环单元、长短期记…...

Midjourney绘画参数设置详解

在数字艺术和设计领域&#xff0c;Midjourney是一款强大的绘画工具&#xff0c;它允许艺术家和设计师以数字方式创作出精美的图像。为了充分发挥Midjourney的潜力&#xff0c;正确设置其绘画参数至关重要。本文将深入探讨Midjourney的绘画参数设置&#xff0c;帮助用户更好地掌…...

计算机毕业设计 | springboot养老院管理系统 老人社区管理(附源码)

1&#xff0c;绪论 1.1 背景调研 养老院是集医疗、护理、康复、膳食、社工等服务服务于一体的综合行养老院&#xff0c;经过我们前期的调查&#xff0c;院方大部分工作采用手工操作方式,会带来工作效率过低&#xff0c;运营成本过大的问题。 院方可用合理的较少投入取得更好…...

事务与并发控制

事务&#xff08;Transaction0&#xff09;&#xff1a;要么全做&#xff0c;要么全不做&#xff1b; 事务ACID&#xff1a;原子性Atomicity&#xff1b;一致性Consistency&#xff1b;隔离性Isolation&#xff1b;持久性Durability&#xff1b; 并发操作问题&#xff1a; 1.…...

spring boot 中的异步@Async

spring boot 开启异步调用 1、启动类上添加EnableAsync注解&#xff0c;表示启动异步 2、在具体实现异步的方法上添加Async注解 package com.example.demo;import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootAppli…...