C++简单实现vector
向量
向量是序列容器,表示可以更改大小的数组。
就像数组一样,向量对其元素使用连续的存储位置,这意味着也可以使用指向其元素的常规指针上的偏移量来访问其元素,并且与数组一样高效。但与数组不同的是,它们的大小可以动态变化,它们的存储由容器自动处理。
在内部,向量使用动态分配的数组来存储其元素。可能需要重新分配此数组,以便在插入新元素时增加大小,这意味着分配新数组并将所有元素移动到该数组。就处理时间而言,这是一项相对昂贵的任务,因此,每次将元素添加到容器时,向量都不会重新分配。
相反,向量容器可以分配一些额外的存储以适应可能的增长,因此容器的实际容量可能大于严格需要的存储来包含其元素(即其大小)。库可以实现不同的增长策略,以平衡内存使用和重新分配,但无论如何,重新分配应仅以对数增长的大小间隔发生,以便可以在向量末尾插入单个元素,并提供摊销的恒定时间复杂性。
因此,与数组相比,向量消耗更多的内存,以换取管理存储和以有效方式动态增长的能力。
与其他动态序列容器(deques、list 和 forward_lists)相比,向量非常有效地访问其元素(就像数组一样),并且相对有效地从其末尾添加或删除元素。对于涉及在末尾以外的位置插入或删除元素的操作,它们的性能比其他元素差,并且迭代器和引用的一致性低于 lists 和 forward_lists。
成员函数
(构造函数) 构造 vector(公开成员函数)
(析构函数) 析构 vector(公开成员函数)
operator= 赋值给容器(公开成员函数)
assign 将值赋给容器(公开成员函数)
get_allocator 返回相关的分配器(公开成员函数)
元素访问
at 访问指定的元素,同时进行越界检查(公开成员函数)
operator[] 访问指定的元素(公开成员函数)
front 访问第一个元素(公开成员函数)
back 访问最后一个元素(公开成员函数)
data 直接访问底层数组(公开成员函数)
迭代器
begin,cbegin(C++11) 返回指向起始的迭代器(公开成员函数)
end,cend(C++11) 返回指向末尾的迭代器(公开成员函数)
rbegin,crbegin(C++11) 返回指向起始的逆向迭代器(公开成员函数)
rend,crend(C++11) 返回指向末尾的逆向迭代器(公开成员函数)
容量
empty 检查容器是否为空(公开成员函数)
size 返回容纳的元素数(公开成员函数)
max_size 返回可容纳的最大元素数(公开成员函数)
reserve 预留存储空间(公开成员函数)
capacity 返回当前存储空间能够容纳的元素数(公开成员函数)
shrink_to_fit(C++11) 通过释放未使用的内存减少内存的使用(公开成员函数)
修改器
clear 清除内容(公开成员函数)
insert 插入元素(公开成员函数)
emplace(C++11) 原位构造元素(公开成员函数)
erase 擦除元素(公开成员函数)
push_back 将元素添加到容器末尾(公开成员函数)
emplace_back(C++11) 在容器末尾就地构造元素(公开成员函数)
pop_back 移除末元素(公开成员函数)
resize 改变容器中可存储元素的个数(公开成员函数)
swap 交换内容(公开成员函数)
非成员函数
按照字典顺序比较 vector 中的值(函数模板)
operator==
operator!=(C++20 中移除)
operator<(C++20 中移除)
operator<=(C++20 中移除)
operator>(C++20 中移除)
operator>=(C++20 中移除)
operator<=>(C++20)
std::swap(std::vector) 特化 std::swap 算法(函数模板)
erase(std::vector),erase_if(std::vector) (C++20) 擦除所有满足特定判别标准的元素(函数模板
cpp
template <typename T>
class Vector
{
public:
Vector() noexcept = default;
explicit Vector(size_t n) : cap_{n}, ptr_{alloc(cap_)}
{
for (; len_ < n; ++len_)
{
construct(ptr_ + len_); //调用T的默认构造
}
}
Vector(size_t n, const T &x) : cap_{n}, ptr_{alloc(cap_)}
{
for (; len_ < n; ++len_)
{
construct(ptr_ + len_, x); //调用T的拷贝构造
}
}
Vector(const Vector &x) : cap_{x.size()}, ptr_{alloc(cap_)} //拷贝构造
{
for (; len_ < x.size(); ++len_)
{
construct(ptr_ + len_, x[len_]);
}
}
Vector(Vector &&x) noexcept //移动构造
{
cap_ = std::__exchange(x.cap_, 0);
len_ = std::__exchange(x.len_, 0);
ptr_ = std::__exchange(x.ptr_, nullptr);
}
Vector(std::initializer_list<T> li) : cap_{li.size()}, ptr_{alloc(cap_)} //初始化列表
{
for (auto &x : li)
{
construct(ptr_ + len_, x);
++len_;
}
}
~Vector() noexcept
{
clear();
dealloc(ptr_);
}
void swap(Vector &x) noexcept
{
using std::swap; // ADL
swap(cap_, x.cap_);
swap(len_, x.len_);
swap(ptr_, x.ptr_);
}
void clear() noexcept
{
for (; len_ > 0; --len_)
{
destroy(ptr_ + len_ - 1);
}
}
Vector &operator=(const T &x) //拷贝赋值
{
if (this != &x)
{
Vector{x}.swap(*this);
}
return *this;
}
Vector &operator=(T &&x) noexcept //移动赋值
{
if (this != &x)
{
Vector{std::move(x)}.swap(*this);
}
return *this;
}
Vector &operator=(std::initializer_list<T> li) //初始化列表赋值
{
Vector{li}.swap(*this);
return *this;
}
void push_back(const T &x) //拷贝
{
emplace_back(x);
}
void push_back(T &&x) //移动
{
emplace_back(x);
}
template <typename... Args>
void emplace_back(Args &&...args) //直接传递构造函数
{
if (len_ == cap_)
{
size_t new_cap = cap_ ? cap_ * 2 : 1; //等0返回1
T *new_ptr = alloc(new_cap);
for (size_t new_len; new_len < len_; ++new_len)
{
construct(new_ptr + new_len, std::move_if_noexcept(ptr_[new_len]));
}
cap_ = new_cap;
ptr_ = new_ptr;
}
construct(ptr_ + len_, std::forward<Args>(args)...);
++len_;
}
void pop_back() noexcept
{
if (len_ < cap_ / 2)
{
size_t new_cap = cap_ / 2;
T *new_ptr = alloc(new_cap);
for (size_t new_len = 0; new_len < len_; ++new_len)
{
construct(new_ptr + new_len, std::move_if_noexcept(ptr_[new_len]));
}
cap_ = new_cap;
ptr_ = new_ptr;
}
destroy(ptr_ + len_ - 1);
--len_;
}
size_t size() const noexcept
{
return len_;
}
size_t capacity() const noexcept
{
return cap_;
}
bool empty() const noexcept
{
return len_ == 0;
}
T &operator[](size_t i)
{
return ptr_[i];
}
const T &operator[](size_t i) const
{
return ptr_[i];
}
T *begin() noexcept
{
return ptr_;
}
T *end() noexcept
{
return ptr_ + len_;
}
const T *begin() const noexcept
{
return ptr_;
}
const T *end() const noexcept
{
return ptr_ + len_;
}
private:
T *alloc(size_t n) //分配n个大小内存
{
return static_cast<T *>(::operator new(sizeof(T) * n));
}
void dealloc(T *p) noexcept //释放内存
{
::operator delete(p);
}
template <typename... Args>
void construct(T *p, Args &&...args) //在这块内存上构造T类型对象
{
::new (p) T(std::forward<Args>(args)...);
}
void destroy(T *p) noexcept
{
p->~T();
}
private:
size_t cap_{0}; //容量
size_t len_{0}; //元素个数
T *ptr_{nullptr};
};
C++简单实现vector的更多相关文章
- 转载 从最简单的vector中sort用法到自定义比较函数comp后对结构体排序的sort算法
转载自:http://www.cnblogs.com/cj695/p/3863142.html sort函数在使用中非常好用,也非常简单,而且效率与冒泡或者选择排序不是一个数量级.本文就sort函数在 ...
- 【转】 从最简单的vector中sort用法到自定义比较函数comp后对结构体排序的sort算法
sort函数在使用中非常好用,也非常简单,而且效率与冒泡或者选择排序不是一个数量级.本文就sort函数在vector中的用法分为sort函数入门用法与自定义comp比较函数比较结构体这两个最基本的功能 ...
- 从最简单的vector中sort用法到自定义比较函数comp后对结构体排序的sort算法
sort函数在使用中非常好用,也非常简单,而且效率与冒泡或者选择排序不是一个数量级.本文就sort函数在vector中的用法分为sort函数入门用法与自定义comp比较函数比较结构体这两个最基本的功能 ...
- 【C++】从最简单的vector中sort用法到自定义比较函数comp后对结构体排序的sort算法
sort函数在使用中非常好用,也非常简单,而且效率与冒泡或者选择排序不是一个数量级.本文就sort函数在vector中的用法分为sort函数入门用法与自定义comp比较函数比较结构体这两个最基本的功能 ...
- C++ STL的简单应用(vector容器专题)
#include <iostream> #include <string> #include <stdlib.h> #include <vector> ...
- 自己动手实现简单的Vector
看到今天,终于自己动手写了一个自己的vector,我这个版本的vector只有vector主要的一些操作,包括原版vector的所有构造函数,begin(),end(),size(),capacity ...
- C++中STL中简单的Vector的实现
该vector只能容纳标准库中string类, 直接上代码了,StrVec.h文件内容为: #ifndef STRVEC_H #define STRVEC_H #include<iostream ...
- vc++简单的vector动态数组实现
#ifndef __MYVECTOR__ #define __MYVECTOR__ #include <Windows.h> #define SUCCESS 1 // 成功 #define ...
- 简单的 vector
#pragma once #include <memory.h> #include <stdlib.h> #include <iostream> using std ...
- C++线性序列容器<vector>简单总结
C++线性序列容器<vector>简单总结 vector是一个长度可变的数组,使用的时候无须声明上限,随着元素的增加,Vector的长度会自动增加:Vector类提供额外的方法来增加.删除 ...
随机推荐
- Java并发编程实例--11.在线程组中处理未检查异常
第8个例子讲了如何在线程中捕捉未检查异常,本例将介绍如何在线程组中处理未检查异常. Task.java package com.dylan.thread.ch1.c11.task; import ja ...
- Spring源码之spring事务
目录 Spring事务 事务自定义标签 自定义标签 解析标签 bean 的初始化 InfrastructureAdvisorAutoProxyCreator 获取增强方法 获取所有增强中内适用于当前方 ...
- windows 上 cmake 添加 vcpkg 选项
使用 cmake 编写相关的工程时,工程有时会使用 vcpkg 添加的第三方库,比如 zip 库 查看一些案例后,我发现有些回答不太准确,遂记录下 现在,我们需要在工程中使用 zip_open 函数执 ...
- macOS使用CodeRunner快速配置fortran环境
个人网站:xzajyjs.cn 由于一些项目的缘故,需要有fortran的需求,但由于是M1 mac的缘故,不能像windows那样直接使用vs+ivf这种经典配置.搜了一下网上主流的跨平台方案,主要 ...
- OpenCV开发笔记(六十八):红胖子8分钟带你使用特征点Flann最邻近差值匹配识别(图文并茂+浅显易懂+程序源码)
若该文为原创文章,未经允许不得转载原博主博客地址:https://blog.csdn.net/qq21497936原博主博客导航:https://blog.csdn.net/qq21497936/ar ...
- LibModbus库开发笔记(一):libmodbus库介绍、编译和基础工程模板
前言 本文章讲解libmodbus. libModbus介绍 libmodbus是一个免费软件库,可根据Modbus协议发送/接收数据.该库用C编写,并支持RTU(串行)和TCP(以太网) ...
- rpartition和partition按分割符分割
# rpartition 从目标字符串的末尾也就是右边开始搜索分割符,如果字符串包含指定的分割符 则返回一个3元的元组,第一个为分割符左边的子串,第二个为分割符本身, 第三个为分割符右边的字串. st ...
- 面向对象基础---day02
成员变量和局部变量区别 封装 private关键字 1.是一个权限修饰符 2.可以修饰成员(成员变量和成员方法) 3.作用是保护成员不被别的类使用,被private修饰的成员只在本类中才能访问 针对p ...
- spark读取和处理zip、gzip、excel、等各种文件最全的技巧总结
一.当后缀名为zip.gzip,spark可以自动处理和读取 1.spark非常智能,如果一批压缩的zip和gzip文件,并且里面为一堆text文件时,可以用如下方式读取或者获取读取后的schema ...
- nginx流量复制与放大
1. 需求 功能需求 在不影响真实业务前提下,支持: 流量复制,用于线故障分析.系统迁移评估等 流量放大,通过多倍复制,实现放大流量,用于性能压测 配置需求 支持或禁止post请求复制 记录镜像请求的 ...