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

【C++干货铺】剖析string | 底层实现

=========================================================================

个人主页点击直达:小白不是程序媛

C++专栏:C++干货铺

代码仓库:Gitee

=========================================================================

目录

成员变量

成员函数

构造和拷贝构造

赋值重载

析构函数

operator[ ]

size

迭代器 

reserve(扩容函数)

push_back(尾插函数)

append(尾插一个字符串)

pos位置插入字符

pos位置插入字符串

删除pos位置的n个字符

rsize

find(查找字符和查找子串)

substr(获取子串)

clear(清除数据)

其他的操作符重载


成员变量


clsss string
{private:char* _str;size_t _size;size_t _capacity;public:const static size_t npos;
}

string实际上是一个字符类型顺序表,因此需要动态开辟空间。_str是指向动态开辟的空间_size用来表示有效数据的个数_capacity表示容量


成员函数

构造和拷贝构造

string(const char * str=""):_size(strlen(str)),_capacity(_size){_str = new char[_capacity+1];strcpy(_str, str);}string(const string& s){_str = new char[s._capacity + 1];strcpy(_str, s._str);_capacity = s._capacity;_size = s._size;}

全缺省有参构造即可以实现有参构造,又可以实现无参构造。使用无参构造时,这里的str什么都不用放,里面面默认含有一个'\0'.这里使用初始化列表,要注意初始化的顺序和成员变量的顺序相同。

赋值重载

	String& operator=(const String& s){if (this != &s){char* tmp = new char[s._capacity + 1];strcpy(tmp, s._str);delete[] _str;_str = tmp;_size = s._size;_capacity = s._capacity;}return* this;}

这里不可以直接对_str重新开辟空间,否则会造成内存泄漏,需要一个中间变量

析构函数

		~string(){delete _str;_str = nullptr;_size = 0;_capacity = 0;}

因为要动态内存开辟,所以要手动释放内存。

operator[ ]

char& operator[](size_t pos){return _str[pos];}

返回字符的引用,用来读写字符。

	const char& operator[](size_t pos)const{return _str[pos];}

返回字符串的引用,const修饰适用于静态创建的对象,只读不可写

size

	size_t size()const{return _size;}

配合operator[ ]可以实现,对一个对象的读写。

迭代器 

    typedef char* iterator;typedef const char* c_iterator;c_iterator begin()const{return _str;}c_iterator end()const{return _str + _size;}iterator begin(){return _str;}iterator end(){return _str + _size;}

beginend函数分别返回字符串的头指针尾指针,配合循环实现迭代器的读和写。

reserve(扩容函数)

	void reserve(size_t n){if (n > _capacity){char* tmp = new char[n + 1];strcpy(tmp, _str);delete[] _str;_str = tmp;_capacity = n;}}

当传入的值大于容量时进行扩容,不可以直接对_str扩容要使用中间变量,防止内存泄漏。

push_back(尾插函数)

	void push_back(char a){//先判断容量满没满if (_capacity == _size){reserve(_capacity == 0 ? 4 : _capacity * 2);}_str[_size] = a;_size++;_str[_size] = '\0';}

通过_size_capacity判断容量是否满了,满了的话调用扩容函数。这里要注意如果是一个空对象,进行尾插时,_capacity为0要对_capacity使用三目操作符判断。并且要在尾插结束后加入'\0',因为字符串的结尾要为'\0'

append(尾插一个字符串)

	void append(const char* str){size_t len=strlen(str);if (len + _size > _capacity){reserve(len + _size);}strcpy(_str + _size, str);_size += len;}

求出插入字符串的长度和有效数据相加判断容量是否足够,不够的话调用reverse函数扩容。

最后在尾指针的位置开始将插入的字符串拷贝进去。

pos位置插入字符

	void insert(size_t pos, char ch){if (_size == _capacity){reserve(_capacity == 0 ? 4 : _capacity * 2);}int end = _size;while (end >= (int)pos){_str[end + 1] = _str[end];end--;}_str[pos] = ch;_size++;}

进行数据的写入一定要判断容量是否足够,移动字符时会发生整形提升造成死循环,要将size_t类型的pos强转成int类型 。移动结束后在pos位置插入字符,修改有效数据的个数。

pos位置插入字符串

	void insert(size_t pos, const char* str){size_t len = strlen(str);if (len + _size > _capacity){reserve(_size + len);}int end = _size;while (end >= (int)pos){_str[end + len] = _str[end];--end;}strncpy(_str + pos, str,len);_size += len;}

和尾插字符串差不多,但是在拷贝时从pos位置拷贝len个字符。最后修改有效数据的个数。

删除pos位置的n个字符

	void erase(size_t pos, size_t len = npos){if (len == npos || pos + len > _size){_size = pos;_str[_size] = '\0';}else{int begin = pos+len;while (begin <= _size){_str[begin-len] = _str[begin];begin++;}_size = _size - len;}}

这是一个全缺省函数,当传入删除的长度时便表示删除从pos位置开始的所有数据。 就直接将有效数据修改为pos,将pos位置的值置为'\0'。

rsize

	void rsize(size_t n, char ch = '\0'){if (n <= _size){_str[n] = '\0';_size = n;}else{reserve(n);while (_size < n){_str[_size] = ch;_size++;}_str[_size] = '\0';}}

半缺省函数,当n小于有效数据时相当于删除有效数据。当n大于有效数据时候先开辟空间在循环设置字符。

find(查找字符和查找子串)

	size_t find(char ch, size_t pos = 0){for (size_t i = pos; i < _size; i++){if (_str[i] == ch){return i;}}return npos;}size_t find(const char* sub, size_t pos = 0){const char* p = strstr(_str + pos, sub);if (p){return p - _str;}else{return npos;}}

对于字符的查找使用循环遍历查找;对于字符串使用strstr库函数查找,在使用指针相加得到位置;

substr(获取子串)

	string substr(size_t pos, size_t len = npos){string s;size_t end = pos + len;if (len == npos || pos + len >= _size){len = _size - pos;end = _size;}s.reserve(len);for (size_t i = pos; i < end; i++){s += _str[i];}return s;}

这里也要判断获取的长度,当len未输入值时候表示取到结尾。创建一个新的对象,将获取的每个字符 存到新的对象中,最后返回该对象。

clear(清除数据)

	void clear(){_str[0] = '\0';_size = 0;}

将有效数据的个数设置成0即可,并不用释放空间


其他的操作符重载

	//+=重载 相当于尾插String& operator +=(char ch){push_back(ch);return *this;}String& operator += (const char* str){append(str);return *this;}bool operator<(const String& s)const{return strcmp(_str, s._str) < 0;}bool operator==(const String& s)const{return strcmp(_str, s._str) == 0;}bool operator<=(const String& s){return (*this < s) || (*this == s);}bool operator>(const String& s){return !(*this <= s);}bool operator>=(const String& s){return !(*this < s);}bool operator<(const String& s){return !(*this >= s);}bool operator!=(const String& s){return !(*this == s);}

今天对string的底层模拟实现的分享到这就结束了,希望大家读完后有很大的收获,也可以在评论区点评文章中的内容和分享自己的看法。您三连的支持就是我前进的动力,感谢大家的支持!!!

相关文章:

【C++干货铺】剖析string | 底层实现

个人主页点击直达&#xff1a;小白不是程序媛 C专栏&#xff1a;C干货铺 代码仓库&#xff1a;Gitee 目录 成员变量 成员函数 构造和拷贝构造 赋值重载 析构函数 operator[ ] size 迭代器 reserve&#xff08;扩容函数&#xff09; push_back(尾插函数&#xff09…...

nmap原理与使用

kali的命令行中可以直接使用 nmap 命令&#xff0c;打开一个「终端」&#xff0c;输入 nmap 后回车&#xff0c;可以看到 nmap 的版本&#xff0c;证明 nmap 可用。 一、端口扫描 扫描主机的「开放端口」&#xff0c;在nmap后面直接跟主机IP&#xff08;默认扫描1000个端口&am…...

AI批量剪辑矩阵托管系统----源码技术开发

AI批量剪辑矩阵托管系统----源码技术开发 抖音账号矩阵系统是基于抖音开放平台研发的用于管理和运营多个抖音账号的平台。它可以帮助用户管理账号、发布内容、营销推广、分析数据等多项任务&#xff0c;从而提高账号的曝光度和影响力。 具体来说&#xff0c;抖音账号矩阵系统可…...

Pandas数据预处理python 数据分析之4——pandas 预处理在线闯关_头歌实践教学平台

Pandas数据预处理python 数据分析之4——pandas 预处理 第1关 数据读取与合并第2关 数据清洗第3关 数据转换 第1关 数据读取与合并 任务描述 本关任务&#xff1a;加载 csv 数据集&#xff0c;实现 DataFrame 合并。 编程要求 根据提示&#xff0c;在右侧编辑器补充代码&#…...

[html] 动态炫彩渐变背景

废话不多说&#xff0c;直接上源码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>ZXW-NUDT: 动态炫…...

AI 绘画 | Stable Diffusion 高清修复、细节优化

前言 在 Stable Diffusion 想要生成高清分辨率的图片。在文生图的功能里&#xff0c;需要设置更大的宽度和高度。在图生图的功能里&#xff0c;需要设置更大的重绘尺寸或者重绘尺寸。但是设置完更大的图像分辨率&#xff0c;需要更大显存&#xff0c;1024*1024的至少要电脑的空…...

想要检测TikTok网络是否安全?这五个网站请收好

Tiktok目前在海外大火&#xff0c;越来越多的人想要进入TikTok的海外市场并捞一桶金。然而&#xff0c;成功并非易事。想要在TikTok中立足&#xff0c;我们必须保证我们的设备、网络环境和网络节点完全符合官方的要求&#xff0c;并且没有任何异常或风险。那么我们该如何设置、…...

【docker:容器提交成镜像】

容器创建部分请看&#xff1a;点击此处查看我的另一篇文章 容器提交为镜像 docker commit -a "sinwa lee" -m "首页变化" mynginx lxhnginx:1.0docker run -d -p 88:80 --name lxhnginx lxhnginx:1.0为啥没有变啊&#xff0c;首页&#xff1f; 镜像打包 …...

UE5中一机一码功能

创建蓝图函数库 1、获取第一个有效的硬盘ID // Fill out your copyright notice in the Description page of Project Settings.#pragma once#include "CoreMinimal.h" #include "Kismet/BlueprintFunctionLibrary.h" #include "GetDiskIDClass.gen…...

gpt支持json格式的数据返回(response_format: ‘json_object‘)

Api.h5.chatCreateChatCompletion({model: gpt-3.5-turbo-1106,token: sk-f4fe8b67-fcbe-46fd-8cc9-fd1dac5d6d59,messages: [{role: user,content:使用json格式返回十二生肖&#xff0c;包含中文名和英文名&#xff0c;[{id:"1", enName:"", cnName: &quo…...

MySQL(13):约束

约束(constraint)概述 数据完整性&#xff08;Data Integrity&#xff09;是指数据的精确性&#xff08;Accuracy&#xff09;和可靠性&#xff08;Reliability&#xff09;。 它是防止数据库中存在不符合语义规定的数据和防止因错误信息的输入输出造成无效操作或错误信息 而提…...

可以为一个servlet定义多个servlet-mapping、或url-pattern

在web描述符文件web.xml文件中&#xff0c;可以为同一个servlet定义多个servlet-mapping&#xff1b;也可以在同一个servlet-mapping中&#xff0c;定义多个url-pattern。也就是说&#xff0c;可以把多个地址&#xff08;相对于上下文路径&#xff09;映射到同一个servlet处理。…...

.net在使用存储过程中IN参数的拼接方案,使用Join()方法

有时候拼接SQL语句时&#xff0c;可能会需要将list中的元素都加上单引号&#xff0c;并以逗号分开&#xff0c;但是Join只能简单的分开&#xff0c;没有有单引号&#xff01; 1.第一种拼接方案 List<string> arrIds new List<string>(); arrIds.Add("aa&qu…...

基于RK3399的室内健身魔镜方案

I 方案背景 一、健身魔镜的兴起 2020年疫情席卷全球&#xff0c;宅家是防疫的措施之一&#xff0c;因而宅家运动火爆&#xff0c;随之而来的宅家运动器材也风靡起来&#xff0c;其中包含既有颜值又具有多种功能的健身魔镜。 Ⅱ 方案介绍 一、健身魔镜的方案介绍 …...

leetCode 25.K 个一组翻转链表

给你链表的头节点 head &#xff0c;每 k 个节点一组进行翻转&#xff0c;请你返回修改后的链表。k 是一个正整数&#xff0c;它的值小于 或 等于链表的长度。如果节点总数不是 k 的整数倍&#xff0c;那么请将最后剩余的节点保持原有顺序。你不能只是单纯的改变节点内部的值&a…...

ElasticSearch中常见的分词器介绍

文章目录 ElasticSearch中常见的分词器介绍前言分词器的作用如何指定分词器分词器的组成分词器的类型标准分词器空格分词器简单分词器关键词分词器停用词分词器IK分词器NGram分词器正则匹配分词器语言分词器自定义分词器 ElasticSearch中常见的分词器介绍 前言 ElasticSearch是…...

前端案例-css实现ul中对li进行换行

场景描述&#xff1a; 我想要实现&#xff0c;在展示的item个数少于4个的时候&#xff0c;则排成一行&#xff0c;并且均分&#xff08;比如说有3个&#xff0c;则每个的宽度为33.3%&#xff09;&#xff0c;如果item 个数大于4&#xff0c;则进行换行。 效果如下&#xff1a…...

【Unity】 场景优化策略

Unity 场景优化策略 GPU instancing 使用GPU Instancing可以将多个网格相同、材质相同、材质属性可以不同的物体合并为一个批次&#xff0c;从而减少Draw Calls的次数。这可以提高性能和渲染效率。 GPU instancing可用于绘制在场景中多次出现的几何体&#xff0c;例如树木或…...

JavaWeb Day09 Mybatis-基础操作01-增删改查

目录 环境准备 ①Emp.sql ②Emp.java 一、删除 ①Mapper层 ②测试类 ③预编译SQL&#xff08;查看mybatis日志&#xff09; 1.性能 2.安全 ④总结 二、新增 ①Mapper层 ②测试类 ③结果 ④新增&#xff08;主键返回&#xff09; 1.Mapper层 2.测试类 ⑤总结​…...

2.前端调试(控制台使用)

消息堆叠 如果一条消息连续重复&#xff0c;而不是在新行上输出每一个消息实例&#xff0c;控制台将“堆叠”消息并在左侧外边距显示一个数字。此数字表示该消息已重复的次数。 如果您倾向于为每一个日志使用一个独特的行条目&#xff0c;请在 DevTools 设置中启用 Show times…...

Python逆向工程实战:如何从pyinstaller打包的.exe文件中找回丢失的源码(附工具包)

Python逆向工程实战&#xff1a;从PyInstaller打包的.exe文件中找回丢失的源码 当你辛苦编写的Python代码因为各种原因丢失&#xff0c;只剩下一个由PyInstaller打包的.exe文件时&#xff0c;那种绝望感我深有体会。作为一名经历过多次类似困境的开发者&#xff0c;我想分享一套…...

嵌入式工程师面试通关指南:从基础理论到实战调试的30个核心考点

1. 嵌入式系统基础概念 1.1 单片机与微处理器的本质区别 很多刚入门的工程师容易混淆单片机和微处理器的概念。简单来说&#xff0c;单片机就是"片上系统"&#xff0c;它把CPU、存储器、I/O接口等核心部件都集成在了一个芯片里。我在设计智能家居控制器时就深有体会…...

架桥记:耐达讯自动化CC-Link IE转EtherCAT的工业协议融合实战

在工业自动化行业中&#xff0c;生产线的智能化升级常面临一个核心难题&#xff1a;如何让基于不同通信协议的设备“读懂”彼此&#xff0c;协同工作&#xff1f;特别是当代表日系高速网络技术的CC-Link IE&#xff0c;遇上盛行于欧系设备的实时以太网EtherCAT时&#xff0c;协…...

new AbortController()

new AbortController() 是 JavaScript 中用于‌创建一个 AbortController 实例‌的构造函数&#xff0c;该实例可用于‌控制或取消异步操作‌&#xff08;如网络请求、事件监听、定时任务等&#xff09;。...

本地AI聊天、交互助手(写给小白的LLM工具选型系列:第三篇)

诸神缄默不语-个人技术博文与视频目录 在这一章介绍的是&#xff0c;已经有了AI大模型推理服务&#xff08;不管是云端API还是本地服务&#xff09;&#xff0c;想要一个像聊天框那样的界面来跟大模型聊天、或者让大模型做更复杂的工作。 本章主要考虑的功能还是AI对话&#x…...

音频转换神器fre:ac:让无损音乐管理更简单

音频转换神器fre:ac&#xff1a;让无损音乐管理更简单 【免费下载链接】freac The fre:ac audio converter project 项目地址: https://gitcode.com/gh_mirrors/fr/freac 价值定位&#xff1a;为什么选择fre:ac作为你的音频处理中心&#xff1f; 在数字音乐时代&#x…...

知识竞赛软件背后的技术架构:从抢答到计分

引言&#xff1a;数字化竞赛的技术基石在现代教育与企业活动中&#xff0c;知识竞赛已成为激发学习热情、检验培训成果的重要形式。一场流畅、公平且充满激情的线上或线下竞赛&#xff0c;其背后离不开一套复杂而精巧的软件系统支撑。这套系统不仅需要提供友好的用户界面&#…...

基于vue3与pinia构建电商核心模块,快马平台实战演练购物车与商品列表

基于vue3与pinia构建电商核心模块&#xff0c;快马平台实战演练购物车与商品列表 最近在做一个电商项目&#xff0c;需要快速搭建商品展示和购物车功能。经过一番调研&#xff0c;我选择了Vue3 Pinia的组合&#xff0c;配合Vue Router实现页面跳转。整个过程在InsCode(快马)平…...

手把手教你:如何根据微软官方文档修改bat脚本,实现Excel文件格式的任意批量转换

从XLS到XLSX&#xff1a;基于微软官方文档的批处理脚本进阶指南 在数据处理工作中&#xff0c;Excel文件格式转换是常见的需求场景。许多用户可能只满足于使用现成的转换工具&#xff0c;但对于技术爱好者或需要频繁处理不同格式转换的专业人士来说&#xff0c;掌握如何根据微…...

从一次jar包热修复踩坑,聊聊Spring Boot的可执行jar原理

从一次jar包热修复踩坑&#xff0c;聊聊Spring Boot的可执行jar原理 那天下午&#xff0c;服务器突然告警&#xff0c;线上服务开始频繁报错。排查后发现是MyBatis的一个XML映射文件存在逻辑缺陷&#xff0c;导致数据库查询结果异常。按照常规流程&#xff0c;本应该修改代码后…...