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

(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类不是为了去实现他&#xff0c;而是为了了解他内部成员函数的一些运行原理和时间复杂度&#xff0c;在将来我们使用时能够合理地去使用他们。 为了避免我们模拟实现的string类与全局上的string类冲突(string类也在std命名空间中)&…...

处理数据中的缺失值--删除缺少值的行

两个最主要的处理缺失值的方法是&#xff1a; ❏ 删除缺少值的行&#xff1b; ❏ 填充缺失值&#xff1b; 我们首先将serum_insulin的中的字段值0替换为None&#xff0c;可以看到缺失值的数量为374个&#xff1b; print(pima[serum_insulin].isnull().sum()) pima[serum_insu…...

Kotlin学习——kt里的集合,Map的各种方法之String篇

Kotlin 是一门现代但已成熟的编程语言&#xff0c;旨在让开发人员更幸福快乐。 它简洁、安全、可与 Java 及其他语言互操作&#xff0c;并提供了多种方式在多个平台间复用代码&#xff0c;以实现高效编程。 https://play.kotlinlang.org/byExample/01_introduction/02_Functio…...

MIT 6.824 -- MapReduce Lab

MIT 6.824 -- MapReduce Lab 环境准备实验背景实验要求测试说明流程说明 实验实现GoLand 配置代码实现对象介绍协调器启动工作线程启动Map阶段分配任务执行任务 Reduce 阶段分配任务执行任务 终止阶段 崩溃恢复 注意事项并发安全文件转换golang 知识点 测试 环境准备 从官方gi…...

创新研报|顺应全球数字化,能源企业以“双碳”为目标的转型迫在眉睫

能源行业现状及痛点分析 挑战一&#xff1a;数字感知能力较弱 挑战二&#xff1a;与业务的融合度低 挑战三&#xff1a;决策响应速度滞后 挑战四&#xff1a;价值创造有待提升 挑战五&#xff1a;安全风险如影随形 能源数字化转型定义及架构 能源行业数字化转型体系大体…...

Blender 连续 5 天遭受大规模 DDoS 攻击

Blender 发布公告指出&#xff0c;在2023年11月18日至23日期间&#xff0c;blender.org 网站遭受了持续的分布式拒绝服务&#xff08;DDoS&#xff09;攻击&#xff0c;攻击者通过不断发送请求导致服务器超载&#xff0c;使网站运营严重中断。此次攻击涉及数百个 IP 地址的僵尸…...

Python 获取本地和广域网 IP

Python 获取本地IP &#xff0c;使用第三方库&#xff0c;比如 netifaces import netifaces as nidef get_ip_address():try:# 获取默认网络接口&#xff08;通常是 eth0 或 en0&#xff09;default_interface ni.gateways()[default][ni.AF_INET][1]# 获取指定网络接口的IP地…...

静态路由配置过程

静态路由 静态路由简介 路由器在转发数据时&#xff0c;要先在路由表&#xff08;Routing Table&#xff09;中在找相应的路由&#xff0c;才能知道数据包应该从哪个端口转发出去。路由器建立路由表基本上有以下三种途径。 &#xff08;1&#xff09;直连路由&#xff1a;路由…...

基于OGG实现MySQL实时同步

&#x1f4e2;&#x1f4e2;&#x1f4e2;&#x1f4e3;&#x1f4e3;&#x1f4e3; 哈喽&#xff01;大家好&#xff0c;我是【IT邦德】&#xff0c;江湖人称jeames007&#xff0c;10余年DBA及大数据工作经验 一位上进心十足的【大数据领域博主】&#xff01;&#x1f61c;&am…...

【计算机网络笔记】多路访问控制(MAC)协议——轮转访问MAC协议

系列文章目录 什么是计算机网络&#xff1f; 什么是网络协议&#xff1f; 计算机网络的结构 数据交换之电路交换 数据交换之报文交换和分组交换 分组交换 vs 电路交换 计算机网络性能&#xff08;1&#xff09;——速率、带宽、延迟 计算机网络性能&#xff08;2&#xff09;…...

什么是好的FPGA编码风格?(3)--尽量不要使用锁存器Latch

前言 在FPGA设计中&#xff0c;几乎没人会主动使用锁存器Latch&#xff0c;但有时候不知不觉中你的设计莫名其妙地就生成了一堆Latch&#xff0c;而这些Latch可能会给你带来巨大的麻烦。 什么是锁存器Latch&#xff1f; Latch&#xff0c;锁存器&#xff0c;一种可以存储电路…...

从0开始学习JavaScript--构建强大的JavaScript图片库

在现代Web开发中&#xff0c;图像是不可或缺的一部分&#xff0c;而构建一个强大的JavaScript图片库能够有效地管理、展示和操作图像&#xff0c;为用户提供更丰富的视觉体验。本文将深入探讨构建JavaScript图片库的实用技巧&#xff0c;并通过丰富的示例代码演示如何实现各种功…...

linux复习笔记05(小滴课堂)

hell脚本与crontab定时器的运用 查看状态&#xff1a; 关闭服务&#xff1a; 开启服务&#xff1a; 重启服务&#xff1a; crontab定时器的使用&#xff1a; 我们可以看到没有任何任务。 编辑&#xff1a; 我们可以看到这个任务了。 删除所有任务&#xff1a; 这代表着每分钟…...

springboot函数式web

1.通常是路由(请求路径)业务 2.函数式web&#xff1a;路由和业务分离 一个configure类 配置bean 路由等 实现业务逻辑 这样实现了业务和路由的分离...

常见的1/2/3位数码管接线详解

今天玩数码管的时候接触到了数码管的接线&#xff0c;分享一下供刚开始接触的童鞋参考 首先了解什么是数码管 数码管是一种可以显示数字和其他信息的电子设备&#xff0c;是显示屏其中一类&#xff0c; 通过对其不同的管脚输入相对的电流&#xff0c;会使其发亮&#xff0c;从而…...

C++模板介绍

定义 C模板是一种编程技术&#xff0c;它允许程序员在编译时生成具有特定类型的函数或类&#xff0c;而无需在运行时进行类型检查。模板是一种泛型编程的方式&#xff0c;它使得程序员可以编写可适用于多种数据类型的代码&#xff0c;提高了代码的重用性和灵活性。 C模板可以…...

kafka kraft 集群搭建保姆级教学 包含几个踩坑点

一.为啥弃用zookeeper kafka 弃用 ZooKeeper 而采用 KRaft 的主要原因是为了改进 Kafka 集群的可靠性和可管理性。 在传统的 Kafka 架构中&#xff0c;ZooKeeper 用于存储和管理集群的元数据、配置信息和状态。然而&#xff0c;使用 ZooKeeper 作为协调服务存在一些限制和挑战…...

html实现360度产品预览(附源码)

文章目录 1.设计来源1.1 拖动汽车产品旋转1.2 汽车产品自动控制 2.效果和源码2.1 动态效果2.2 源代码 源码下载 作者&#xff1a;xcLeigh 文章地址&#xff1a;https://blog.csdn.net/weixin_43151418/article/details/134613931 html实现360度产品预览&#xff08;附源码&…...

11-23 SSM4

Ajax 同步请求 &#xff1a;全局刷新的方式 -> synchronous请求 客户端发一个请求&#xff0c;服务器响应之后你客户端才能继续后续操作&#xff0c;请求二响应完之后才能发送后续的请求&#xff0c;依次类推 有点&#xff1a;服务器负载较小&#xff0c;但是由于服务器相应…...

CPU、GPU、TPU内存子系统架构

文章目录 CPU、GPU、TPU内存子系统架构概要CPUGPUTPU共同点和差异&#xff1a; CPU、GPU、TPU内存子系统架构 概要 Memory Subsystem Architecture&#xff0c;图源自TVM CPU CPU&#xff08;中央处理器&#xff09;的内存子系统&#xff1a;隐式管理 主内存&#xff08;…...

ArduPilot SITL不止能飞Copter:手把手教你用同一套环境玩转无人机、固定翼和无人车仿真

ArduPilot SITL全平台仿真指南&#xff1a;从无人机到无人车的无缝切换 当你第一次成功运行ArduPilot的多旋翼无人机仿真时&#xff0c;那种兴奋感可能还记忆犹新。但你知道吗&#xff1f;你刚刚搭建的这套Ubuntu 22.04环境&#xff0c;其实是一把能打开整个无人系统世界的万能…...

基于Phi-4-mini-reasoning的智能数据匹配方案:告别VLOOKUP跨表烦恼

基于Phi-4-mini-reasoning的智能数据匹配方案&#xff1a;告别VLOOKUP跨表烦恼 1. 场景痛点&#xff1a;VLOOKUP的跨表匹配困境 财务小王最近遇到了一个头疼的问题&#xff1a;每个月末都要处理几十张报表的数据匹配工作。这些数据分散在不同工作簿中&#xff0c;需要根据产品…...

使用Spring AI Alibaba构建智能体Agent妒

背景 在软件开发的漫长旅途中&#xff0c;"构建"这个词往往让人又爱又恨。爱的是&#xff0c;一键点击&#xff0c;代码变成产品&#xff0c;那是程序员最迷人的时刻&#xff1b;恨的是&#xff0c;维护那一堆乱糟糟的构建脚本&#xff0c;简直是噩梦。 在很多项目中…...

AI时代的算法思维:大经典排序学习拐

引言 在现代软件开发中&#xff0c;性能始终是衡量应用质量的重要指标之一。无论是企业级应用、云服务还是桌面程序&#xff0c;性能优化都能显著提升用户体验、降低基础设施成本并增强系统的可扩展性。对于使用 C# 开发的应用程序而言&#xff0c;性能优化涉及多个层面&#x…...

【HTML动态交互实战】模拟股市波动可视化系统

1. 从零搭建股市波动可视化系统 最近在做一个金融数据分析的小项目&#xff0c;需要模拟股票价格波动并可视化展示。作为一个前端开发者&#xff0c;我第一时间想到用HTML5 Canvas来实现这个需求。下面就把我的实现思路和踩过的坑分享给大家。 先说说为什么要用Canvas而不是S…...

植物大战僵尸修改器:3分钟解锁无限游戏乐趣的终极指南

植物大战僵尸修改器&#xff1a;3分钟解锁无限游戏乐趣的终极指南 【免费下载链接】pvztoolkit 植物大战僵尸 PC 版综合修改器 项目地址: https://gitcode.com/gh_mirrors/pv/pvztoolkit 还在为阳光不够用而烦恼&#xff1f;无尽模式卡在第10波就过不去&#xff1f;想保…...

多模态入门新选择:ViLT模型实战,从文本处理到图像理解的统一Transformer玩法

多模态入门新选择&#xff1a;ViLT模型实战&#xff0c;从文本处理到图像理解的统一Transformer玩法 当你第一次听说多模态学习时&#xff0c;脑海中可能会浮现出复杂的双流架构、繁琐的区域特征提取&#xff0c;以及让人望而生畏的计算资源需求。这正是大多数Vision-and-Langu…...

SqlHelper 使用手册

目录 一、核心方法概览 二、ExecuteNonQuery - 增删改操作 常用示例 重载形式 三、事务支持 四、ExecuteDataset - 查询数据集 五、ExecuteReader - 流式读取 六、ExecuteScalar - 获取单值 七、SqlHelperParameterCache - 参数缓存 八、参数传递方式对比 九、快速参…...

如何在3分钟内完成Windows与Office智能激活:KMS_VL_ALL_AIO完整指南

如何在3分钟内完成Windows与Office智能激活&#xff1a;KMS_VL_ALL_AIO完整指南 【免费下载链接】KMS_VL_ALL_AIO Smart Activation Script 项目地址: https://gitcode.com/gh_mirrors/km/KMS_VL_ALL_AIO 你是否曾经因为Windows或Office的激活问题而烦恼&#xff1f;当系…...

STM32F103片内Flash 存储器操作(FLASH页划分)

一、Flash 基础1.1 什么是 Flash&#xff1f;Flash Memory&#xff1a; 闪存存储器定义&#xff1a; 一种非易失性存储器&#xff0c;掉电后数据不丢失核心特性&#xff1a;非易失性&#xff1a; 掉电后数据不丢失 可擦写&#xff1a; 可以多次擦除和写入 块操作&#xff1a; 擦…...