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

【C++】——string(模拟实现)

文章目录

  • string类构造
  • string类拷贝构造
  • string类析构
  • string类运算符重载
  • string类部分常用接口的模拟实现

这篇博客中构造、拷贝构造、析构、还有一些短小频繁调用的函数就不用做函数和定义分离,因为在类中,这种函数会默认是内联函数

string类构造

构造函数有一些容易忽略的细节,我会在下面的代码注释中说明

		#include<iostream>
#include<assert.h>
#include<string.h>
using namespace std;
namespace sg
{class string{public:string()//:_str(nullptr) // 不能这么写,因为当我string对象为空,程序解引用到\0才会终止,遇不到就会崩溃:_str(new char[1]{'\0'}) // 空间里至少要有一个字符, _size(0), _capacity(0){}//构造函数string(const char* str){_size = strlen(str);_capacity = strlen(str);_str = new char[strlen(str) + 1];strcpy(_str, str);}private:char* _str;size_t _size;size_t _capacity;};void test_string1();}

还有另一种写法,不用初始化,直接在构造函数的参数给上缺省值

/*string():_str(new char[1]{'\0'}), _size(0), _capacity(0)
{}*/
//构造函数
//string(string(const char* str = nullptr)
// 如果参数又给空,程序还是会崩,C语言规定常量字符串后面都会有\0,所以直接给空字符串就行
string(const char* str = "")
{_size = strlen(str);_capacity = strlen(str);_str = new char[strlen(str) + 1];strcpy(_str, str);
}

string类拷贝构造

如果使用默认拷贝构造,会引发浅拷贝问题,就是当两个对象指向同一块资源时,释放资源后,会导致一块资源被析构两次,程序会报错。所以自己写一个深拷贝最好

//拷贝构造函数
//s2(s1)
string(const string& s)// this就是s2
{_str = new char[s._capacity + 1]; // 开一个和s1一样大的空间strcpy(_str, s._str); _size = s._size;_capacity = s._capacity;
}

string类析构

string的析构和c语言栈实现中的销毁栈大差不差,如果有疑问可以参考这一篇博客
C语言栈和队列

//析构函数
~string()
{if (_str)     //检查一下_str是否为空,如果为空就不用再释放空间了{delete[] _str;_str = nullptr;_size = _capacity = 0;}    }

string类运算符重载

string.h

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

string.cpp

#define   _CRT_SECURE_NO_WARNINGS 1
#include "practice-string.h"
namespace sg
{void test_string1(){string s1;string s2("hello world");cout << s1.c_str() << endl;cout << s2.c_str() << endl;for (size_t i = 0; i < s2.size(); i++){s2[i] += 2;}for (auto e : s2){cout << e << " ";}cout << endl;cout << s2.c_str() << endl;}
}

我通过测试,发现并不能用范围for遍历,报错显示未找到匹配的begin函数,需要我们自己实现
范围for看着很方便,但本质就是迭代器,迭代器模拟的是指针的行为,所以我们返回原生指针即可

string.h

typedef char* iterator;
typedef 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;
}

赋值运算符

string operator=(const string& s)
{if (this != &s)//地址相同,就是自己赋值给自己{delete[] _str;_str = new char[s._capacity + 1];strcpy(_str, s._str);_size = s._size;_capacity = s._capacity;}return *this;
}

string类部分常用接口的模拟实现

string.h

#pragma once
#include<iostream>
#include<assert.h>
#include<string.h>
using namespace std;
namespace sg
{class string{public:typedef char* iterator;typedef 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;}/*string():_str(new char[1]{'\0'}), _size(0), _capacity(0){}*///构造函数string(const char* str = ""){_size = strlen(str);_capacity = strlen(str);_str = new char[strlen(str) + 1];strcpy(_str, str);}//析构函数~string(){delete[] _str;_str = nullptr;_size = _capacity = 0;}//拷贝构造函数//s2(s1)string(const string& s){_str = new char[s._capacity + 1];strcpy(_str, s._str);_size = s._size;_capacity = s._capacity;}const char* c_str() const{return _str;}size_t size() const{return _size;}char& operator[](size_t pos){assert(pos < _size);return _str[pos];}const char& operator[](size_t pos) const{assert(pos < _size);return _str[pos];}// s2 = s1// s1 = s1string operator=(const string& s){if (this != &s)//地址相同,就是自己赋值给自己{delete[] _str;_str = new char[s._capacity + 1];strcpy(_str, s._str);_size = s._size;_capacity = s._capacity;}return *this;}void clear(){_str[0] = '\0';_size = 0;}void reserve(size_t n);void push_back(char ch);void append(const char* str);string& operator+=(char ch);string& operator+=(const char* str);void insert(size_t pos, char ch);void insert(size_t pos, const char* str);void erase(size_t pos, size_t len = npos);size_t find(char ch, size_t pos = 0);size_t find(const char* str, size_t pos = 0);string substr(size_t pos = 0, size_t len = npos);private:char* _str;size_t _size;size_t _capacity;static const size_t npos;};void test_string1();void test_string2();bool operator<(const string& s1, const string& s2);bool operator<=(const string& s1, const string& s2);bool operator>(const string& s1, const string& s2);bool operator>=(const string& s1, const string& s2);bool operator!=(const string& s1, const string& s2);bool operator==(const string& s1, const string& s2);ostream& operator<<(ostream& out, const string& s);istream& operator>>(istream& in, string& s);
}

string.cpp

#define   _CRT_SECURE_NO_WARNINGS 1
#include"string.h"
namespace sg
{const size_t string::npos = -1;//声明和定义分离void string::reserve(size_t n){if (n > _capacity){char* tmp = new char[n + 1];strcpy(tmp, _str);delete[] _str;_str = tmp;_capacity = n;}}void string::push_back(char ch){if (_size == _capacity){reserve(_capacity == 0 ? 4 : _capacity * 2);}_str[_size] = ch;++_size;_str[_size] = '\0';}string& string::operator+=(char ch){push_back(ch);return *this;}void string::append(const char* str){size_t len = strlen(str);if (_size + len > _capacity){//大于二倍,需要多少开多少,小于二倍,就按二倍扩reserve(_size + len > 2 * _capacity ? _size + len : 2 * _capacity);}strcpy(_str + _size, str);//从\0开始拷贝,\0会自动被挤到后面_size += len;}string& string::operator+=(const char* str){append(str);return *this;}void string::insert(size_t pos, char ch){assert(pos < _size);if (_size == _capacity){reserve(_capacity == 0 ? 4 : _capacity * 2);}//挪动数据/*size_t end = _size;while (end >= pos){_str[end + 1] = _str[end];--end;}*///挪动数据size_t end = _size + 1;while (end > pos){_str[end] = _str[end - 1];--end;}//挪动完数据后再插入_str[pos] = ch;++_size;}void string::insert(size_t pos, const char* str){assert(pos < _size);if (_size == _capacity){reserve(_capacity == 0 ? 4 : _capacity * 2);}size_t len = strlen(str);size_t end = _size + len;while (end >= pos + len){_str[end] = _str[end - len];--end;}for (size_t i = 0; i < len; i++){_str[pos + i] = str[i];}_size += len;}void string::erase(size_t pos, size_t len){if (len >= _size - pos){_str[pos] = '\0';_size = pos;}else{for (size_t i = pos + len; i <= _size; i++){_str[i - len] = _str[i];}_size -= len;}}size_t string::find(char ch, size_t pos){assert(pos < _size);for (size_t i = pos; i < _size; i++){if (_str[i] == ch){return i;}}return npos;}size_t string::find(const char* str, size_t pos){assert(pos < _size);const char* ptr = strstr(_str + pos, str);if (ptr == nullptr){return npos;}else{return ptr - _str;}}string string::substr(size_t pos = 0, size_t len){assert(pos < _size);if (len > _size - pos){len = _size - pos;}string sub;sub.reserve(len);for (size_t i = 0; i < len; i++){sub += _str[pos + i];}return sub;}bool operator<(const string& s1, const string& s2){return strcmp(s1.c_str(), s2.c_str()) < 0;}bool operator<=(const string& s1, const string& s2){return s1 == s2 || s1 < s2;}bool operator>(const string& s1, const string& s2){return !(s1 <= s2);}bool operator>=(const string& s1, const string& s2){return !(s1 < s2);}bool operator!=(const string& s1, const string& s2){return !(s1 == s2);}bool operator==(const string& s1, const string& s2){return strcmp(s1.c_str(), s2.c_str()) == 0;}ostream& operator<<(ostream& out, const string& s){for (auto ch : s){out << ch;}return out;}istream& operator>>(istream& in, string& s){s.clear();//创造一块缓冲const int N = 256;char buffer[N];int i = 0;char ch;ch = in.get();while (ch != ' ' && ch != '\0'){buffer[i++] == ch;if (i == N - 1)// buffer条件{buffer[i] = '\0';s += buffer;i = 0;}if (i > 0){buffer[i] = '\0';s += buffer;}return in;}}void test_string1(){string s1;string s2("hello world");cout << s1.c_str() << endl;cout << s2.c_str() << endl;for (size_t i = 0; i < s2.size(); i++){s2[i] += 2;}cout << s2.c_str() << endl;}void test_string2(){string s1("hello world");s1 += '!';s1 += '#';cout << s1.c_str() << endl;s1 += "renqing";cout << s1.c_str() << endl;s1.insert(0, '#');cout << s1.c_str() << endl;string s2("hello world");s2.insert(5, "big");cout << s2.c_str() << endl;s2.erase(5, 3);cout << s2.c_str() << endl;}}

test.cpp

#define   _CRT_SECURE_NO_WARNINGS 1
#include"string.h"
int main()
{//sg::test_string1();sg::test_string2();return 0;
}

相关文章:

【C++】——string(模拟实现)

文章目录 string类构造string类拷贝构造string类析构string类运算符重载string类部分常用接口的模拟实现 这篇博客中构造、拷贝构造、析构、还有一些短小频繁调用的函数就不用做函数和定义分离&#xff0c;因为在类中&#xff0c;这种函数会默认是内联函数 string类构造 构造函…...

c++20 std::format 格式化说明

在标头<format>定义 ()功能很强大&#xff0c;它把字符串当成一个模板&#xff0c;通过传入的参数进行格式化&#xff0c;并且使用大括号‘{}’作为特殊字符代替‘%’。 1、基本用法 &#xff08;1&#xff09;不带编号&#xff0c;即“{}”&#xff08;2&#xff09;带…...

HTB-Unified(log4j2漏洞、MongoDb替换管理员密码)

前言 各位师傅大家好&#xff0c;我是qmx_07&#xff0c;今天给大家讲解Unified靶机 渗透过程 信息搜集 服务器开放了SSH服务&#xff0c;HTTP服务 访问网站 验证log4j2漏洞 8443端口&#xff1a;UniFi 网络 &#xff0c;访问查询 是否有Nday漏洞利用 可以观察到UniFi的版…...

每天五分钟深度学习PyTorch:不同的神经网络层设置不同的学习率

本文重点 我们前面学习了基本网络模型的搭建,获取网络模型的子结构,以及优化器optim,我们发现我们设置优化器的时候,是对整个模型设置的,也就是说整个模型的参数学习率是一样,本节课程我们学习如何给不同的网络层设置不同的学习率。主要还是通过优化器optim来实现的,本…...

【渗透测试】——DVWA靶场搭建

&#x1f4d6; 前言&#xff1a;DVWA&#xff08;Damn Vulnerable Web Application&#xff09;是一个用于安全漏洞测试的 PHP/MySQL 网络应用&#xff0c;旨在为安全专业人士提供一个合法的环境&#xff0c;以测试他们的技能和工具&#xff0c;同时帮助 Web 开发者更好地理解 …...

国内人工智能产业发展现状及对策研究

一、引言 人工智能作为新时代科技革命和产业变革的核心力量&#xff0c;正深刻改变着全球经济格局。我国政府高度重视人工智能产业发展&#xff0c;将其列为国家战略性新兴产业。在此背景下&#xff0c;本文旨在分析我国人工智能产业发展现状&#xff0c;探讨面临的挑战&#x…...

完整版订单超时自动取消功能

前几天对实习还是继续学习技术产生了抉择&#xff0c;问了一个前辈&#xff0c;他抛给我一个问题&#xff0c;怎么做15分钟订单自动取消&#xff0c;我说然后到时间之后&#xff0c;自动执行这个订单关闭业务&#xff0c;比如把锁了的库存给解开等等操作&#xff0c;然后在数据…...

算法刷题:300. 最长递增子序列、674. 最长连续递增序列、718. 最长重复子数组、1143. 最长公共子序列

300. 最长递增子序列 1.dp定义&#xff1a;dp[i]表示i之前包括i的以nums[i]结尾的最长递增子序列的长度 2.递推公式&#xff1a;if (nums[i] > nums[j]) dp[i] max(dp[i], dp[j] 1); 注意这里不是要dp[i] 与 dp[j] 1进行比较&#xff0c;而是我们要取dp[j] 1的最大值…...

go 笔记

数据结构与 方法&#xff08;增删改查&#xff09; 安装goland,注意版本是2024.1.1&#xff0c;不是2024.2.1&#xff0c;软件下载地址也在链接中提供了 ‘go’ 不是内部或外部命令&#xff0c;也不是可运行的程序 或批处理文件。 在 Windows 搜索栏中输入“环境变量”&#…...

路由等保测评

1.身份鉴别 应对登录的用户进行身份标识和鉴别&#xff0c; 身份标识具有唯一性&#xff0c;身份鉴别信息具有复杂度要求并定期更换。 可以使用“ service password-encryption"命令对存储在配置文件中的所有口令和类似数据进行加密&#xff0c; 以避免攻击者通过读取配…...

C# 反射之动态生成dll/exe

这个可能应该属于反射的高级使用范围了&#xff0c;平常在项目中使用的人估计也不是很多。由于使用反射的话会降低性能&#xff0c;比如之前用到的GetValue、SetValue等之类&#xff0c;但是使用这种方式会大大提高效率&#xff0c;在这里我只想说&#xff0c;都直接写IL指令了…...

Rust 所有权 Slices

文章目录 发现宝藏1. Slice 的基础知识1.1 什么是 Slice&#xff1f;1.2 如何创建 Slice&#xff1f; 2. 处理字符串 Slice2.1 字符串的 Slice2.2 字符串的 Unicode 和切片 3. 在函数中使用 Slice3.1 传递 Slice 给函数3.2 可变 Slice 的函数 4. 复杂示例4.1 处理多维数组的 Sl…...

windows 安全与网络管理问题

问题&#xff1a;当编写的脚本或程序运行的时候&#xff0c;可能被windows阻止访问网络甚至被删除 避免被删除 wini 进入设置界面 -> 选择更新与安全 -> 选择windwos defender -> 点击添加排除项&#xff0c;将指定的文件或目录排除&#xff0c;避免被软件删除 允许…...

基于Python实现一个庆祝国庆节的小程序

功能&#xff1a; 添加互动功能&#xff1a;允许用户选择不同的祝福语或者查询不同的国庆节信息。动态背景音乐&#xff1a;播放国庆节相关的背景音乐。增加节日小测验&#xff1a;提供一些关于国庆节的趣味小测验&#xff0c;让用户参与。增强图形用户界面 (GUI)&#xff1a;…...

Anaconda 安装与使用教程

Anaconda 安装与使用教程 介绍 Anaconda 是一个用于科学计算的 Python 和 R 的发行版&#xff0c;它包含了众多流行的科学计算、数据分析、机器学习等领域的库。本教程旨在帮助初学者快速上手 Anaconda&#xff0c;并学会如何使用其管理环境以及安装包。 第一步&#xff1a;…...

时序预测SARIMAX模型

1. 项目背景 本文基于kaggle平台相关竞赛项目&#xff0c;具体连接如下&#xff1a; Time Series Forecasting With SARIMAX 基本信息如内容说明、数据集、已提交代码、当前得分排名以及比赛规则等&#xff0c;如图【1】所示&#xff0c;可以认真阅读。 图 1 2. 数据读取 …...

gin集成jaeger中间件实现链路追踪

1. 背景 新业务线带来新项目启动&#xff0c;需要改进原有项目的基础框架和组件能力&#xff0c;以提升后续开发和维护效率。项目搭建主要包括技术选型、框架搭建、基础服务搭建等。这其中就涉及到链路追踪的内容&#xff0c;结合其中的踩坑情况&#xff0c;用一篇文章来说明完…...

前端层面----监控与埋点

前言&#xff1a; 站在产品的视角&#xff0c;经常会问如下几个问题&#xff1a; 产品有没有用户使用 用户用得怎么样 系统会不会经常出现异常 如何更好地满足用户需求服务用户 当站在技术视角时&#xff0c;经常会问如下几个问题&#xff1a; 系统出现异常的频率如何 异常…...

linux Command

linux Command 1. 系统监控命令 1.1 top top [param] top -H -p pid&#xff0c;查看进程pid下面的子线程。-b以处理模式操作-c显示完整的命令行而不只是显示命令名。-d 屏幕刷新间隔时间。-l 忽略失效过程。-s 保密模式。-S 累积模式。-u 【用户名】 指定用户名。-p 【进程…...

uniapp登录页面( 适配:pc、小程序、h5)

<!-- 简洁登录页面 --> <template><view class"login-bg"><image class"img-a" src"https://zhoukaiwen.com/img/loginImg/2.png"></image><image class"img-b" src"https://zhoukaiwen.com/im…...

3秒破解百度网盘提取码:提升资源获取效率的自动化工具指南

3秒破解百度网盘提取码&#xff1a;提升资源获取效率的自动化工具指南 【免费下载链接】baidupankey 项目地址: https://gitcode.com/gh_mirrors/ba/baidupankey 场景痛点&#xff1a;被提取码困住的数字生活 你是否经历过这样的场景&#xff1a;深夜赶项目时&#xf…...

GLM-OCR模型长短期记忆(LSTM)解码器技术剖析

GLM-OCR模型长短期记忆&#xff08;LSTM&#xff09;解码器技术剖析 你是不是觉得现在的OCR&#xff08;光学字符识别&#xff09;技术特别神奇&#xff1f;一张图片拍下去&#xff0c;里面的文字瞬间就被识别出来了&#xff0c;又快又准。这背后&#xff0c;除了强大的视觉模…...

物联网新手避坑指南:用MQTT.fx 1.7.1连接OneNET平台,从数据上报到命令下发的完整流程

物联网开发实战&#xff1a;MQTT.fx与OneNET平台深度对接指南 第一次打开MQTT.fx时&#xff0c;面对密密麻麻的配置项和晦涩的协议术语&#xff0c;大多数物联网初学者都会感到无从下手。这就像刚拿到驾照就让你开F1赛车——工具很强大&#xff0c;但学习曲线陡峭。本文将带你用…...

OpenClaw实操指南09|云端部署实战:腾讯云+OpenClaw,打造7×24小时不断线AI助手

很多人第一次用OpenClaw&#xff0c;是在自己电脑上跑的。 用着挺爽——但只要关机&#xff0c;AI助手就断了。出门在路上&#xff0c;飞书消息发出去&#xff0c;没有回应。 本地部署的致命缺陷&#xff1a;你不在&#xff0c;它也不在。 这篇教程解决这个问题。用腾讯云轻…...

PCB丝印设计规范与Altium高效调整技巧

1. PCB丝印调整的必要性与基本原则 在PCB设计流程中&#xff0c;丝印位号的调整往往被初学者视为"收尾工作"而草率处理。但作为有十年Layout经验的工程师&#xff0c;我必须强调&#xff1a;规范的丝印设计能提升30%以上的装配效率&#xff0c;特别是对于需要手工焊接…...

微信小程序物流查询插件接入全攻略:从资质申请到waybill_token获取(附完整代码)

微信小程序物流查询插件深度接入指南&#xff1a;全流程解析与实战代码 最近在帮一个电商客户优化小程序时&#xff0c;发现物流查询功能直接影响了30%的用户留存率。微信官方提供的物流查询插件确实能解决这个问题&#xff0c;但接入过程中遇到的坑比想象中多得多。今天就把完…...

别只用自带库了!LTspice仿真进阶:手把手教你搭建带运放的‘智能’稳压电路,性能对比一目了然

别只用自带库了&#xff01;LTspice仿真进阶&#xff1a;手把手教你搭建带运放的‘智能’稳压电路&#xff0c;性能对比一目了然 在电子设计领域&#xff0c;稳压电路就像是为系统提供稳定"血液"的心脏。许多工程师习惯使用LTspice自带库中的齐纳二极管快速搭建基础稳…...

2026年留学生essay降AI怎么做?绕开3个坑,Turnitin检测轻松过

留学生用AI写essay是普遍现象&#xff0c;但Turnitin的AI检测越来越准&#xff0c;被抓到的后果很严重。问题是&#xff0c;降AI这件事里有好几个坑&#xff0c;踩中了哪怕用最好的工具也可能不通过。 先说结论&#xff1a;这3个坑是最常见的&#xff0c;绕开之后&#xff0c;…...

0欧姆电阻在电子设计中的关键应用与选型指南

1. 0欧姆电阻的实质与特性在电子工程实践中&#xff0c;0欧姆电阻&#xff08;Zero-Ohm Resistor&#xff09;是一种表面贴装或插装形式的特殊电子元件。虽然标称值为零欧姆&#xff0c;但实际测量时会发现其存在微小的阻值——典型值在20-50毫欧之间。这个特性使其既不同于理想…...

人工改写和降AI工具哪个更划算?花了一周测完给你结论

这个问题我认真测过&#xff1a;同一篇论文&#xff0c;用人工改写和用降AI工具处理&#xff0c;分别花了多少时间、多少钱、最终效果如何。结论有点出乎意料。 结论&#xff1a;降AI工具性价比远高于人工改写&#xff0c;唯一例外是对语言质量有极高要求的顶刊投稿。推荐的工…...