(C++)string类的模拟实现
愿所有美好如期而遇
前言
我们模拟实现string类不是为了去实现他,而是为了了解他内部成员函数的一些运行原理和时间复杂度,在将来我们使用时能够合理地去使用他们。
为了避免我们模拟实现的string类与全局上的string类冲突(string类也在std命名空间中),我们用自己的命名空间包起来。
namespace A
{class string{static void test(){}}
}int main()
{A::string::test();
}
接下来我们开始模拟实现string类。
私有的成员变量,我们设计无符号整型计算有效字符数量的_size,以及记录空间大小的_capacity,最后就是字符指针,我们将来开辟好一块空间后存放字符串,将由他来指向。
namespace A {class string{private:size_t _size;size_t _capacity;char* _str;} }
构造函数:
默认是空串,并且我们默认比有效字符多开一个字节的空间,之后进行拷贝。
string(const char* str = "") {_size = _capacity = strlen(str);_str = new char[_capacity + 1];strcpy(_str, str); }
拷贝构造函数:
我们模拟实现string类,一定会有空间的开辟,一但对象涉及到拷贝,就不能是浅拷贝,否则会出现一块空间两次释放,直接崩掉。
string(const string& str) {_capacity = str._capacity;_size = str._size;_str = new char[_capacity + 1];strcpy(_str, str.c_str()); }
赋值运算符重载:
同拷贝构造,对象也是不能值拷贝,否则会出现一块空间释放两次。
string& operator=(const string& str) {if (this != &str){_capacity = str._capacity;_size = str._size;char* tmp = new char[_capacity + 1];strcpy(tmp, str.c_str());delete[] _str;_str = tmp;} return *this; }
析构函数:
~string() {delete[] _str;_str = nullptr;_capacity = _size = 0; }
将string转换为const char*,我们会在strlen等C的接口处使用,比如strlen(s.c_str()),因为strlen他的参数是const char*类型的。
const char* c_str() const {return _str; }
四个迭代器
只有物理地址空间连续才可以像这样,如果是链表,我们不能这样做。
typedef char* iterator; typedef const char* const_iterator;iterator begin() {return _str; }iterator end() {return _str + _size; }const_iterator begin()const {return _str; }const_iterator end()const {return _str + _size; }
尾插单个字符,_capacity和_size都是只考虑有效字符,我们在创建空间时默认开空间比_capacity大一。
void push_back(char ch) {if (_size == _capacity){_capacity = 0 ? 4 : _capacity * 2;char* tmp = new char[_capacity + 1];strcpy(tmp, _str);delete[] _str;_str = tmp;}_str[_size++] = ch;_str[_size] = '\0'; }
reserve函数,我们使用reserve重新开空间,也是只考虑有效字符。
我们实现的reserve不可以缩容。
void reserve(size_t n) {if (n > _capacity){char* tmp = _str;//比需求多开一个字节的空间_str = new char[n + 1];strcpy(_str, tmp);//别忘记_capacity_capacity = n;delete[] tmp;} }
尾插字符串
void append(const char* str) {size_t len = strlen(str);if (_size + len > _capacity){reserve(_size + len);}strcpy(_str + _size, str);_size += len; }
赋值运算符+=的重载
//三个版本的+=运算符重载 string& operator+=(const string& str) {append(str.c_str());return *this; }string& operator+=(const char* str) {append(str);return *this; }string& operator+=(char ch) {push_back(ch);return *this; }
成员函数swap
//不用进行深拷贝,直接交换地址和大小 void swap(string& s) {std::swap(_size, s._size);std::swap(_capacity, s._capacity);std::swap(_str, s._str); }
清空字符串函数
void clear() {memset(_str, 0, _size); }返回有效字符数量 ,判断是否为空
size_t size()const {return _size; }bool empty()const {return _size == 0; }
重载[]运算符,我们重载两个版本的,重载const版本是因为我们有时候只读,不写,所以需要这样的重载。
char& operator[](size_t index) {return _str[index]; }const char& operator[](size_t index)const {return _str[index]; }
find函数,找到后返回下标。strstr找到后返回的是找到的字符串的首地址,否则返回空。
size_t find(char c, size_t pos) const {for (size_t i = pos; i < _size; i++){if (_str[i] == c){return i;}}return npos; }//hello world size_t find(const char* s, size_t pos = 0) const {char* tmp = strstr(_str, s);if (tmp == nullptr){return npos;}return tmp - _str; }
insert函数,从pos位置插入单个字符,以及字符串
string& insert(size_t pos, char c) {assert(pos <= _size);if (_size == _capacity){_capacity = 0 ? 4 : _capacity * 2;char* tmp = new char[_capacity + 1];strcpy(tmp, _str);delete[] _str;_str = tmp;}int end = _size;while (end >= (int)pos){_str[end + 1] = _str[end];end--;}_str[pos] = c;_size++;return *this; }//pos是下标 string& insert(size_t pos, const char* str) {assert(pos <= _size);int len = strlen(str);if (len + pos > _capacity){reserve(len + pos);}int end = _size;while (end >= (int)pos){_str[end + len] = _str[end];end--;}strncpy(_str + pos, str, len);_size += len;return *this; }
erase函数,删除从pos节点开始的len个字符
string& erase(size_t pos, size_t len) {//如果pos == size就和没删除一样,相当于删\0assert(pos < _size);//从pos位置删除len个字符,也就是pos后的字符全部删掉//传的len值接近npos,pos+len可能溢出,也会有未知错误if (len == npos || pos + len >= _size){_str[pos] = '\0';_size = pos;}else{strcpy(_str + pos, _str + pos + len);_size -= len;}return *this; }
substr,截取从pos位置开始长度为len的字符串 ,在他返回时,就是对象的值拷贝,会有临时对象产生,临时对象就需要我们自己写的拷贝构造函数了。
string substr(size_t pos = 0, size_t len = npos) {assert(pos < _size);int end;string str;//hello world 2 3if (len == npos || pos + len > _size){end = _size;}else{end = pos + len;}//记得给str扩容str.reserve(end - pos);while (end > pos){str += _str[pos++];}return str; }
相关文章:
(C++)string类的模拟实现
愿所有美好如期而遇 前言 我们模拟实现string类不是为了去实现他,而是为了了解他内部成员函数的一些运行原理和时间复杂度,在将来我们使用时能够合理地去使用他们。 为了避免我们模拟实现的string类与全局上的string类冲突(string类也在std命名空间中)&…...
处理数据中的缺失值--删除缺少值的行
两个最主要的处理缺失值的方法是: ❏ 删除缺少值的行; ❏ 填充缺失值; 我们首先将serum_insulin的中的字段值0替换为None,可以看到缺失值的数量为374个; print(pima[serum_insulin].isnull().sum()) pima[serum_insu…...
Kotlin学习——kt里的集合,Map的各种方法之String篇
Kotlin 是一门现代但已成熟的编程语言,旨在让开发人员更幸福快乐。 它简洁、安全、可与 Java 及其他语言互操作,并提供了多种方式在多个平台间复用代码,以实现高效编程。 https://play.kotlinlang.org/byExample/01_introduction/02_Functio…...
MIT 6.824 -- MapReduce Lab
MIT 6.824 -- MapReduce Lab 环境准备实验背景实验要求测试说明流程说明 实验实现GoLand 配置代码实现对象介绍协调器启动工作线程启动Map阶段分配任务执行任务 Reduce 阶段分配任务执行任务 终止阶段 崩溃恢复 注意事项并发安全文件转换golang 知识点 测试 环境准备 从官方gi…...
创新研报|顺应全球数字化,能源企业以“双碳”为目标的转型迫在眉睫
能源行业现状及痛点分析 挑战一:数字感知能力较弱 挑战二:与业务的融合度低 挑战三:决策响应速度滞后 挑战四:价值创造有待提升 挑战五:安全风险如影随形 能源数字化转型定义及架构 能源行业数字化转型体系大体…...
Blender 连续 5 天遭受大规模 DDoS 攻击
Blender 发布公告指出,在2023年11月18日至23日期间,blender.org 网站遭受了持续的分布式拒绝服务(DDoS)攻击,攻击者通过不断发送请求导致服务器超载,使网站运营严重中断。此次攻击涉及数百个 IP 地址的僵尸…...
Python 获取本地和广域网 IP
Python 获取本地IP ,使用第三方库,比如 netifaces import netifaces as nidef get_ip_address():try:# 获取默认网络接口(通常是 eth0 或 en0)default_interface ni.gateways()[default][ni.AF_INET][1]# 获取指定网络接口的IP地…...
静态路由配置过程
静态路由 静态路由简介 路由器在转发数据时,要先在路由表(Routing Table)中在找相应的路由,才能知道数据包应该从哪个端口转发出去。路由器建立路由表基本上有以下三种途径。 (1)直连路由:路由…...
基于OGG实现MySQL实时同步
📢📢📢📣📣📣 哈喽!大家好,我是【IT邦德】,江湖人称jeames007,10余年DBA及大数据工作经验 一位上进心十足的【大数据领域博主】!😜&am…...
【计算机网络笔记】多路访问控制(MAC)协议——轮转访问MAC协议
系列文章目录 什么是计算机网络? 什么是网络协议? 计算机网络的结构 数据交换之电路交换 数据交换之报文交换和分组交换 分组交换 vs 电路交换 计算机网络性能(1)——速率、带宽、延迟 计算机网络性能(2)…...
什么是好的FPGA编码风格?(3)--尽量不要使用锁存器Latch
前言 在FPGA设计中,几乎没人会主动使用锁存器Latch,但有时候不知不觉中你的设计莫名其妙地就生成了一堆Latch,而这些Latch可能会给你带来巨大的麻烦。 什么是锁存器Latch? Latch,锁存器,一种可以存储电路…...
从0开始学习JavaScript--构建强大的JavaScript图片库
在现代Web开发中,图像是不可或缺的一部分,而构建一个强大的JavaScript图片库能够有效地管理、展示和操作图像,为用户提供更丰富的视觉体验。本文将深入探讨构建JavaScript图片库的实用技巧,并通过丰富的示例代码演示如何实现各种功…...
linux复习笔记05(小滴课堂)
hell脚本与crontab定时器的运用 查看状态: 关闭服务: 开启服务: 重启服务: crontab定时器的使用: 我们可以看到没有任何任务。 编辑: 我们可以看到这个任务了。 删除所有任务: 这代表着每分钟…...
springboot函数式web
1.通常是路由(请求路径)业务 2.函数式web:路由和业务分离 一个configure类 配置bean 路由等 实现业务逻辑 这样实现了业务和路由的分离...
常见的1/2/3位数码管接线详解
今天玩数码管的时候接触到了数码管的接线,分享一下供刚开始接触的童鞋参考 首先了解什么是数码管 数码管是一种可以显示数字和其他信息的电子设备,是显示屏其中一类, 通过对其不同的管脚输入相对的电流,会使其发亮,从而…...
C++模板介绍
定义 C模板是一种编程技术,它允许程序员在编译时生成具有特定类型的函数或类,而无需在运行时进行类型检查。模板是一种泛型编程的方式,它使得程序员可以编写可适用于多种数据类型的代码,提高了代码的重用性和灵活性。 C模板可以…...
kafka kraft 集群搭建保姆级教学 包含几个踩坑点
一.为啥弃用zookeeper kafka 弃用 ZooKeeper 而采用 KRaft 的主要原因是为了改进 Kafka 集群的可靠性和可管理性。 在传统的 Kafka 架构中,ZooKeeper 用于存储和管理集群的元数据、配置信息和状态。然而,使用 ZooKeeper 作为协调服务存在一些限制和挑战…...
html实现360度产品预览(附源码)
文章目录 1.设计来源1.1 拖动汽车产品旋转1.2 汽车产品自动控制 2.效果和源码2.1 动态效果2.2 源代码 源码下载 作者:xcLeigh 文章地址:https://blog.csdn.net/weixin_43151418/article/details/134613931 html实现360度产品预览(附源码&…...
11-23 SSM4
Ajax 同步请求 :全局刷新的方式 -> synchronous请求 客户端发一个请求,服务器响应之后你客户端才能继续后续操作,请求二响应完之后才能发送后续的请求,依次类推 有点:服务器负载较小,但是由于服务器相应…...
CPU、GPU、TPU内存子系统架构
文章目录 CPU、GPU、TPU内存子系统架构概要CPUGPUTPU共同点和差异: CPU、GPU、TPU内存子系统架构 概要 Memory Subsystem Architecture,图源自TVM CPU CPU(中央处理器)的内存子系统:隐式管理 主内存(…...
ArduPilot SITL不止能飞Copter:手把手教你用同一套环境玩转无人机、固定翼和无人车仿真
ArduPilot SITL全平台仿真指南:从无人机到无人车的无缝切换 当你第一次成功运行ArduPilot的多旋翼无人机仿真时,那种兴奋感可能还记忆犹新。但你知道吗?你刚刚搭建的这套Ubuntu 22.04环境,其实是一把能打开整个无人系统世界的万能…...
基于Phi-4-mini-reasoning的智能数据匹配方案:告别VLOOKUP跨表烦恼
基于Phi-4-mini-reasoning的智能数据匹配方案:告别VLOOKUP跨表烦恼 1. 场景痛点:VLOOKUP的跨表匹配困境 财务小王最近遇到了一个头疼的问题:每个月末都要处理几十张报表的数据匹配工作。这些数据分散在不同工作簿中,需要根据产品…...
使用Spring AI Alibaba构建智能体Agent妒
背景 在软件开发的漫长旅途中,"构建"这个词往往让人又爱又恨。爱的是,一键点击,代码变成产品,那是程序员最迷人的时刻;恨的是,维护那一堆乱糟糟的构建脚本,简直是噩梦。 在很多项目中…...
AI时代的算法思维:大经典排序学习拐
引言 在现代软件开发中,性能始终是衡量应用质量的重要指标之一。无论是企业级应用、云服务还是桌面程序,性能优化都能显著提升用户体验、降低基础设施成本并增强系统的可扩展性。对于使用 C# 开发的应用程序而言,性能优化涉及多个层面&#x…...
【HTML动态交互实战】模拟股市波动可视化系统
1. 从零搭建股市波动可视化系统 最近在做一个金融数据分析的小项目,需要模拟股票价格波动并可视化展示。作为一个前端开发者,我第一时间想到用HTML5 Canvas来实现这个需求。下面就把我的实现思路和踩过的坑分享给大家。 先说说为什么要用Canvas而不是S…...
植物大战僵尸修改器:3分钟解锁无限游戏乐趣的终极指南
植物大战僵尸修改器:3分钟解锁无限游戏乐趣的终极指南 【免费下载链接】pvztoolkit 植物大战僵尸 PC 版综合修改器 项目地址: https://gitcode.com/gh_mirrors/pv/pvztoolkit 还在为阳光不够用而烦恼?无尽模式卡在第10波就过不去?想保…...
多模态入门新选择:ViLT模型实战,从文本处理到图像理解的统一Transformer玩法
多模态入门新选择:ViLT模型实战,从文本处理到图像理解的统一Transformer玩法 当你第一次听说多模态学习时,脑海中可能会浮现出复杂的双流架构、繁琐的区域特征提取,以及让人望而生畏的计算资源需求。这正是大多数Vision-and-Langu…...
SqlHelper 使用手册
目录 一、核心方法概览 二、ExecuteNonQuery - 增删改操作 常用示例 重载形式 三、事务支持 四、ExecuteDataset - 查询数据集 五、ExecuteReader - 流式读取 六、ExecuteScalar - 获取单值 七、SqlHelperParameterCache - 参数缓存 八、参数传递方式对比 九、快速参…...
如何在3分钟内完成Windows与Office智能激活:KMS_VL_ALL_AIO完整指南
如何在3分钟内完成Windows与Office智能激活:KMS_VL_ALL_AIO完整指南 【免费下载链接】KMS_VL_ALL_AIO Smart Activation Script 项目地址: https://gitcode.com/gh_mirrors/km/KMS_VL_ALL_AIO 你是否曾经因为Windows或Office的激活问题而烦恼?当系…...
STM32F103片内Flash 存储器操作(FLASH页划分)
一、Flash 基础1.1 什么是 Flash?Flash Memory: 闪存存储器定义: 一种非易失性存储器,掉电后数据不丢失核心特性:非易失性: 掉电后数据不丢失 可擦写: 可以多次擦除和写入 块操作: 擦…...
