在 C++ 中实现自定义容器的实用指南
在 C++ 中实现自定义容器的实用指南
在 C++ 编程中,容器是存储和管理数据的基本工具。标准库提供了多种容器,如 std::vector
、std::list
和 std::map
,但在某些情况下,开发者可能需要实现自定义容器以满足特定需求。本文将详细介绍如何在 C++ 中实现一个简单的自定义容器,包括设计思路、实现细节和使用示例。
一、设计自定义容器
在开始实现自定义容器之前,我们需要明确容器的基本功能和特性。以下是设计自定义容器时需要考虑的几个方面:
- 容器类型:选择容器的类型,例如线性容器(如数组、链表)或关联容器(如哈希表、树)。
- 存储策略:决定如何存储元素,使用动态数组、链表或其他数据结构。
- 基本操作:定义容器支持的基本操作,如插入、删除、查找和遍历。
- 内存管理:考虑如何管理内存,避免内存泄漏和悬空指针。
二、实现一个简单的动态数组容器
我们将实现一个简单的动态数组容器 MyVector
,它支持基本的插入、删除和访问操作。以下是 MyVector
的基本结构:
1. 头文件和类定义
#include <iostream>
#include <stdexcept>template <typename T>
class MyVector {
public:MyVector(); // 构造函数~MyVector(); // 析构函数void push_back(const T& value); // 添加元素void pop_back(); // 删除最后一个元素T& operator[](size_t index); // 访问元素size_t size() const; // 获取元素数量bool empty() const; // 检查容器是否为空private:T* data; // 存储数据的指针size_t capacity; // 容量size_t count; // 当前元素数量void resize(); // 调整容器大小
};
2. 构造函数和析构函数
template <typename T>
MyVector<T>::MyVector() : capacity(2), count(0) {data = new T[capacity]; // 初始化容量为2
}template <typename T>
MyVector<T>::~MyVector() {delete[] data; // 释放内存
}
3. 添加元素
push_back
方法用于在容器末尾添加元素。如果当前容量不足,则调用 resize
方法扩展容量。
template <typename T>
void MyVector<T>::push_back(const T& value) {if (count == capacity) {resize(); // 扩展容量}data[count++] = value; // 添加元素
}template <typename T>
void MyVector<T>::resize() {capacity *= 2; // 容量翻倍T* newData = new T[capacity]; // 创建新数组for (size_t i = 0; i < count; ++i) {newData[i] = data[i]; // 复制旧数据}delete[] data; // 释放旧数组data = newData; // 更新指针
}
4. 删除元素
pop_back
方法用于删除容器末尾的元素。
template <typename T>
void MyVector<T>::pop_back() {if (count == 0) {throw std::out_of_range("Vector is empty"); // 检查是否为空}--count; // 减少元素数量
}
5. 访问元素
重载 operator[]
以支持通过索引访问元素。
template <typename T>
T& MyVector<T>::operator[](size_t index) {if (index >= count) {throw std::out_of_range("Index out of range"); // 检查索引有效性}return data[index]; // 返回元素
}
6. 获取元素数量和检查空
实现 size
和 empty
方法。
template <typename T>
size_t MyVector<T>::size() const {return count; // 返回当前元素数量
}template <typename T>
bool MyVector<T>::empty() const {return count == 0; // 检查是否为空
}
三、使用自定义容器
现在我们已经实现了一个简单的动态数组容器 MyVector
,接下来我们将展示如何使用它。
1. 示例代码
int main() {MyVector<int> vec; // 创建 MyVector 实例// 添加元素vec.push_back(1);vec.push_back(2);vec.push_back(3);// 访问元素for (size_t i = 0; i < vec.size(); ++i) {std::cout << vec[i] << " "; // 输出: 1 2 3}std::cout << std::endl;// 删除最后一个元素vec.pop_back();// 输出当前元素数量std::cout << "Size after pop: " << vec.size() << std::endl; // 输出: 2return 0;
}
2. 编译和运行
将上述代码保存为 main.cpp
,然后使用以下命令编译和运行:
g++ -o my_vector main.cpp
./my_vector
四、总结
在本文中,我们实现了一个简单的动态数组容器 MyVector
,并展示了如何使用它。通过这个示例,我们了解了自定义容器的基本设计思路和实现细节。在实际开发中,自定义容器可以根据特定需求进行扩展和优化,例如支持迭代器、异常安全、移动语义等。
自定义容器的实现不仅能帮助我们更好地理解 C++ 的内存管理和模板编程,还能提高我们解决实际问题的能力。希望本文能为你在 C++ 编程中实现自定义容器提供一些启发和帮助。
相关文章:
在 C++ 中实现自定义容器的实用指南
在 C 中实现自定义容器的实用指南 在 C 编程中,容器是存储和管理数据的基本工具。标准库提供了多种容器,如 std::vector、std::list 和 std::map,但在某些情况下,开发者可能需要实现自定义容器以满足特定需求。本文将详细介绍如何…...
《深入浅出WPF》读书笔记.4名称空间详解
《深入浅出WPF》读书笔记.4名称空间详解 背景 主要讲明名称空间概念,可以理解为命名空间的引用。 xmlns:x"http://schemas.microsoft.com/winfx/2006/xaml" 👆如x可以理解为一些列命名空间的引用。 不一一列举,只讲几个特殊的…...
电驱动总成
电驱动总成(Electric Drive Assembly)是电动汽车和混合动力汽车中关键的组成部分,主要负责将电能转化为机械能,以驱动汽车的轮胎。电驱动总成包括多个关键组件,通常可以分为以下几个主要部分: ### 主要组成…...

JavaScript class和正则
正则表达式练习 出生日期 年 月 日 ()表示一个整体 console.log(1909.match(^19\\d{2}$)); console.log(2024.match(^20(([01][0-9])|(2[0-4]))$)); //年 console.log(1909.match(^(19\\d{2})|(20(([01][0-9])|(2[0-4])))$)); // 月 console.log(12.match(^(0[1-9])|(1[0-2])…...

[Linux#42][线程] 锁的接口 | 原理 | 封装与运用 | 线程安全
互斥量 mutex • 大部分情况,线程使用的数据都是局部变量,变量的地址空间在线程栈空间 内,这种情况,变量归属单个线程,其他线程无法获得这种变量。 • 但有时候,很多变量都需要在线程间共享,这…...
奇异递归Template有啥奇的?
如果一个模版看起来很头痛,那么大概率这种模版是用来炫技,没啥用的,但是CRTP这个模版,虽然看起来头大,但是却经常被端上桌~ 奇异递归模板模式(Curiously Recurring Template Pattern, CRTP)是一…...

每天五分钟深度学习框架pytorch:神经网络工具箱nn的介绍
本文重点 我们前面一章学习了自动求导,这很有用,但是在实际使用中我们基本不会使用,因为这个技术过于底层,我们接下来将学习pytorch中的nn模块,它是构建于autograd之上的神经网络模块,也就是说我们使用pytorch封装好的神经网络层,它自动会具有求导的功能,也就是说这部…...

【办公软件】安全风险 Microsoft 已阻止宏运行,因为此文件的来源不受信任
Excel 2019版本,就出现安全风险 Microsoft 已阻止宏运行 因为此文件的来源不受信任的问题,宏直接就用不了了。 网上的解决方法,文件右键属性->取消安全锁。但存在没有安全锁这个选项。后查询到一个简单的解决方法。 打开Excel表格->文件…...

JavaScript语法基础之流程结构(顺序、选择、循环结构)
目录 1. 流程控制 1.1. 流程控制简介 1.1.1. 顺序结构 1.1.2. 选择结构 1.1.3. 循环结构 1.2. 选择结构:if 1.2.1. 单向选择:if… 1.2.2. 双向选择:if…else… 1.2.3. 多向选择:if…else_if…else… 1.3. 选择结构&#…...

集团数字化转型方案(四)
集团数字化转型方案通过全面部署人工智能(AI)、大数据分析、云计算和物联网(IoT)技术,创建了一个智能化的企业运营平台,涵盖从业务流程自动化、实时数据监控、精准决策支持,到个性化客户服务和高…...
【MySQL索引】索引失效场景
索引失效 1 全值匹配肯定不失效 2 最佳左前缀法则 索引文件具有 B-Tree 的最左前缀匹配特性,如果左边的值未确定,那么无法使用此索引。 3 主键插入顺序 页分裂,建议 让主键具有 AUTO_INCREMENT 4 计算、函数、类型转换(自动或手动)导致…...

基于MATLAB视觉的静态手势识别系统
一、课题介绍及思路 为了丰富手势识别方法的多样性,提高手势识别的正确率,提出了一种基于手势轮廓像素变化的手势识别方法。在Matlab环境下,设计并开发了一个基于视觉的静态手势识别系统。系统主要由两部分组成:手势分割与手势识…...
day02-作业题
一、简答题 请说出方法定义的全格式 访问修饰符 静态修饰符 返回值 方法名(参数列表){方法体;retrun 返回值;}请说出方法重载的概念 在一个类中,可以定义方法名相同,参数列表不相同(参数类型、参数个数或者参数顺序不同)的方法请简述什么是类…...

torch.cuda.set_divice()
我申请了两块GPU,然后看两张显卡的编号 import torch torch.cuda.set_device(0) # 设置当前cuda设备编号为1 print("当前cuda设备是", torch.cuda.current_device()) # 获取当前cuda设备import torch torch.cuda.set_device(1) …...

<数据集>RSOD数据集<目标检测>
数据集格式:VOCYOLO格式 图片数量:936张 标注数量(xml文件个数):936 标注数量(txt文件个数):936 标注类别数:4 标注类别名称:[aircraft, oiltank, overpass, playground] 序号类别名称图片数框数1air…...

企业高性能web服务器之Nginx
文章目录 Apache经典的web服务端Apache prefork 模型Apache work 模型(适应市场)Apache event 模型 网络I/O网络I/O模型I/O模型网络I/O模型 Nginx架构和安装Nginx源码编译环境准备安装nginx Nginx的平滑升级及版本回滚 Nginx架构和进程Nginx进程结构Ngin…...

11-sentinel利用nacos作持久化
本文介绍sentinel配置数据的持久化方法。由于sentinel官方并没有提供持久化功能,大家在测试过程中也能发现sentinel服务重启后,原来配置的数据就丢了,本文就是来处理这一问题的。 做好心理准备,我们要修改sentinel的源代码&#…...
密码学之哈希算法
文章目录 1. 哈希函数概述1.1 哈希函数的定义1.2 哈希函数的重要性 2. SHA系列算法简介2.1 SHA系列的发展历史2.2 SHA系列的应用场景 3. 主要SHA算法详解3.1 MD5算法3.2 SHA-1算法3.3 SHA-2算法家族3.4 SHA-3算法 4. SHA算法的安全性分析4.1 安全性的重要性4.2 已知的攻击方法4…...

杰发科技AC7801——GPIO通过寄存器地址控制高低电平
通过这个寄存器来查看控制的是哪个ODR值,使用sample,发现是0x20080068的第7和第9位 使用51控制寄存器的代码来置高置低代码,注意变量需要用unsigned int来声明 unsigned int ledBit 0;mdelay(100);ledBit | (1 << 9); ledBit & ~…...
代码随想录算法训练营第三十一天| 01背包问题 二维 01背包问题 一维 416. 分割等和子集
01背包问题 二维 代码随想录 视频讲解:带你学透0-1背包问题!| 关于背包问题,你不清楚的地方,这里都讲了!| 动态规划经典问题 | 数据结构与算法_哔哩哔哩_bilibili #include <bits/stdc.h> using namespace std;…...
设计模式和设计原则回顾
设计模式和设计原则回顾 23种设计模式是设计原则的完美体现,设计原则设计原则是设计模式的理论基石, 设计模式 在经典的设计模式分类中(如《设计模式:可复用面向对象软件的基础》一书中),总共有23种设计模式,分为三大类: 一、创建型模式(5种) 1. 单例模式(Sing…...

C++初阶-list的底层
目录 1.std::list实现的所有代码 2.list的简单介绍 2.1实现list的类 2.2_list_iterator的实现 2.2.1_list_iterator实现的原因和好处 2.2.2_list_iterator实现 2.3_list_node的实现 2.3.1. 避免递归的模板依赖 2.3.2. 内存布局一致性 2.3.3. 类型安全的替代方案 2.3.…...
pam_env.so模块配置解析
在PAM(Pluggable Authentication Modules)配置中, /etc/pam.d/su 文件相关配置含义如下: 配置解析 auth required pam_env.so1. 字段分解 字段值说明模块类型auth认证类模块,负责验证用户身份&am…...
蓝桥杯 2024 15届国赛 A组 儿童节快乐
P10576 [蓝桥杯 2024 国 A] 儿童节快乐 题目描述 五彩斑斓的气球在蓝天下悠然飘荡,轻快的音乐在耳边持续回荡,小朋友们手牵着手一同畅快欢笑。在这样一片安乐祥和的氛围下,六一来了。 今天是六一儿童节,小蓝老师为了让大家在节…...

HTML 列表、表格、表单
1 列表标签 作用:布局内容排列整齐的区域 列表分类:无序列表、有序列表、定义列表。 例如: 1.1 无序列表 标签:ul 嵌套 li,ul是无序列表,li是列表条目。 注意事项: ul 标签里面只能包裹 li…...

【2025年】解决Burpsuite抓不到https包的问题
环境:windows11 burpsuite:2025.5 在抓取https网站时,burpsuite抓取不到https数据包,只显示: 解决该问题只需如下三个步骤: 1、浏览器中访问 http://burp 2、下载 CA certificate 证书 3、在设置--隐私与安全--…...

Module Federation 和 Native Federation 的比较
前言 Module Federation 是 Webpack 5 引入的微前端架构方案,允许不同独立构建的应用在运行时动态共享模块。 Native Federation 是 Angular 官方基于 Module Federation 理念实现的专为 Angular 优化的微前端方案。 概念解析 Module Federation (模块联邦) Modul…...
css的定位(position)详解:相对定位 绝对定位 固定定位
在 CSS 中,元素的定位通过 position 属性控制,共有 5 种定位模式:static(静态定位)、relative(相对定位)、absolute(绝对定位)、fixed(固定定位)和…...

在WSL2的Ubuntu镜像中安装Docker
Docker官网链接: https://docs.docker.com/engine/install/ubuntu/ 1、运行以下命令卸载所有冲突的软件包: for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg; done2、设置Docker…...

C# 求圆面积的程序(Program to find area of a circle)
给定半径r,求圆的面积。圆的面积应精确到小数点后5位。 例子: 输入:r 5 输出:78.53982 解释:由于面积 PI * r * r 3.14159265358979323846 * 5 * 5 78.53982,因为我们只保留小数点后 5 位数字。 输…...