【C++教程从0到1入门编程】第八篇:STL中string类的模拟实现
一、 string类的模拟实现
#include <iostream>
namespace y
{class string{public: //string() //无参构造函数// :_str(nullptr)//{}//string(char* str) //有参构造函数// :_str(str)//{}string():_str(new char[1]){_str[0] = '\0';}string(char* str) //构造函数在堆上开辟一段strlen+1的空间+1是c_str:_str(new char[strlen(str)+1]){strcpy(_str, str); //strcpy会拷贝\0过去}//string(char* str="") //构造函数在堆上开辟一段strlen+1的空间+1是c_str// :_str(new char[strlen(str) + 1])//{// strcpy(_str, str); //strcpy会拷贝\0过去//}size_t size(){return strlen(_str);}bool empty(){return _str == nullptr;}char& operator[](size_t i) //用引用返回不仅可以读字符,还可以修改字符{return _str[i];}~string() //析构函数{if (_str){delete[] _str;_str = nullptr;}}const char* c_str() //返回C的格式字符串{return _str;}private:char* _str;};void TestString1(){string s1("hello");string s2;for (size_t i = 0; i < s1.size(); i++){s1[i] += 1;std::cout << s1[i] << " ";}std::cout << std::endl;for (size_t i = 0; i < s2.size(); i++){s2[i] += 1;std::cout << s2[i] << " ";}std::cout << std::endl;}void TestString2(){string s1("hello");string s2(s1);std::cout << s1.c_str() << std::endl;std::cout << s2.c_str() << std::endl;string s3("world");s1 = s3; //调试点这里,析构也是两次std::cout << s1.c_str() << std::endl;std::cout << s3.c_str() << std::endl;}
}
这段代码中,存在一定的问题,当我们调试时会发现!
默认的拷贝的构造函数出现的问题!
!默认的赋值运算符重载出现的问题。!
上述string类没有显式定义其拷贝构造函数与赋值运算符重载,此时编译器会合成默认的,当用s1构造s2时,编译器会调用默认的拷贝构造。最终导致的问题是,s1、s2共用同一块内存空间,在释放时同一块空间被释放多次而引起程序崩溃,这种拷贝方式,称为浅拷贝。
此时引出了概念浅拷贝,
浅拷贝:也称位拷贝,编译器只是将对象中的值拷贝过来。如果对象中管理资源,最后就会导致多个对象共享同一份资源,当一个对象销毁时就会将该资源释放掉,而此时另一些对象不知道该资源已经被释放,以为还有效,所以 当继续对资源进项操作时,就会发生发生了访问违规。要解决浅拷贝问题,C++中引入了深拷贝。
那么深拷贝呢?
二、string类的模拟实现
头文件代码:
#include<iostream>
#include<assert.h>
namespace yyw
{class string{public:typedef char* iterator;public:string(const char* str = "") //构造函数{_size = strlen(str);_capacity = _size;_str = new char[_capacity + 1];strcpy(_str, str);}//string s2(s1)string(const string& s):_str(nullptr), _size(0), _capacity(0){string tmp(s._str);this->swap(tmp);}//s1=s3string& operator=(const string& s) //(string s){//this->swap(s);string tmp(s._str);this->swap(tmp);return *this;}void swap(string& s){std::swap(_str, s._str);std::swap(_size, s._size);std::swap(_capacity, s._capacity);}~string() //析构函数{if (_str){delete[] _str;_size = _capacity = 0;_str = nullptr;}}//string(const string& s) //拷贝构造void push_back(char ch) //增加字符{if (_size == _capacity) //增加空间{size_t newcapacity = _capacity == 0 ? 6 : _capacity * 2;char* tmp = new char[newcapacity + 1];strcpy(tmp, _str);delete[] _str;_str = tmp;_capacity = newcapacity;}_str[_size] = ch;_size++;_str[_size] = '\0'; //_size的位置设置为\0}void append(char* str) //追加字符串{size_t len = strlen(str);if (_size + len > _capacity) //注意不能按2倍去增容{size_t newcapacity = _size + len;char* tmp = new char[newcapacity + 1];strcpy(tmp, _str);delete[] _str;_str = tmp;_capacity = newcapacity;}strcpy(_str + _size, str);_size += len;//_str[_size + len] = '\0'; strcpy已经把\0拷贝过去了}//s1+='ch' s1就是thisstring& operator+=(char ch){this->push_back(ch);return *this;}//s1+="ch" s1就是thisstring& operator+=(char* ch){this->append(ch);return *this;}string& insert(size_t pos, char ch) //在pos位置插入字符{assert(pos <= _size);if (_size == _capacity){size_t newcapacity = _capacity == 0 ? 4 : _capacity * 2;char* tmp = new char[newcapacity + 1];strcpy(tmp, _str);delete[] _str;_str = tmp;//delete[] _str; //注意这里不能写反了_capacity = newcapacity;}size_t end = _size;while (end >= (int)pos){_str[end + 1] = _str[end];end--;}_str[pos] = ch;_size++;return *this;}string& insert(size_t pos, char* str) //在pos位置插入字符串{assert(pos < _size);size_t len = strlen(str);if (_size + len > _capacity){size_t newcapacity = _size + len;char* tmp = new char[newcapacity + 1];strcpy(tmp, _str);delete[] _str;_str = tmp;_capacity = newcapacity;}size_t end = _size;while (end >= (int)pos){_str[end + len] = _str[end]; //这里是挪len个不是1个end--;}//strncpy也可以//strncpy(_str + pos, str, len);//strcpy会把\0拷贝过去,不可以//写个循环从pos依次往后放for (size_t i = 0; i < len; i++){_str[pos] = str[i];pos++;}_size += len;//返回自己return *this;}string& erase(size_t pos, size_t len = npos){assert(pos < _size);if (len >= _size - pos){_str[pos] = '\0';_size = pos;}else{size_t i = pos + len;while (i <= _size){_str[i - len] = _str[i];i++;}_size = _size - len;}return *this;}size_t find(char ch, size_t pos) //在pos位置查找字符{for (size_t i = pos; i < _size; i++){if (_str[i] == ch){return i;}}return npos;}size_t find(char* str, size_t pos) //在pos位置查找字符串{char* p = strstr(_str, str);if (p == NULL){return npos;}else{return (p - str);}}void resize(size_t newsize, char ch = '\0') //填充字符ch{if (newsize < _size) //第三种情况{_str[newsize] = '\0';_size = newsize;}else{if (newsize > _capacity) //增加容量{size_t newcapacity = newsize;char* tmp = new char[newcapacity];strcpy(tmp, _str);delete[]_str;_str = tmp;_capacity = newcapacity;}for (size_t i = _size; i < newsize; i++) //把字符ch往_size后面填{_str[i] = ch;}_size = newsize;_str[_size] = '\0';}}iterator begin() //iterator迭代器的原理{return _str;}iterator end(){return (_str + _size);}const char* c_str(){return _str;}char& operator[](size_t i) //重载[]可以遍历输出字符串,加&是既可以读,也可以写{assert(i < _size);return _str[i];}const char& operator[](size_t i) const{assert(i < _size);return _str[i];}//s1<s s1就是thisbool operator<(const string& s){int ret = strcmp(_str, s._str);return ret < 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){int ret = strcmp(_str, s._str);return ret == 0;}bool operator!=(const string& s){return !(*this == s);}bool empty(){return _size == 0;}size_t size() //求字符串的大小{return _size;}size_t capacity() //求字符串的容量{return _capacity;}private:char* _str;size_t _size; //已经有多少个有效字符个数size_t _capacity; //能存多少个有效字符个数 \0不是有效字符,\0是标识结束的字符static size_t npos; //insert用的位置};size_t string::npos = -1;std::ostream& operator<<(std::ostream& _out, string& s) //重载输出运算符<<{for (size_t i = 0; i < s.size(); i++){std::cout << s[i];}return _out;}std::istream& operator>>(std::istream& _in, string& s) //重载输入运算符<<{//for (size_t i = 0; i < s.size(); i++) 错误写法//{// std::cin >> s[i];//}while (1){char ch;ch = _in.get();if (ch == ' ' || ch == '\n'){break;}else{s += ch;}}return _in;}//std::istream& operator>>(std::istream& _in, string &s) //重载输入运算符<<//{// //for (size_t i = 0; i < s.size(); i++) 错误写法// //{// // std::cin >> s[i];// //}// while (1)// {// char ch;// ch = _in.get();// if ( ch == '\n')// {// break;// }// else// {// s += ch;// }// }// return _in;//}void TestString1(){string s1;string s2("bit");for (size_t i = 0; i < s1.size(); i++){std::cout << s1[i] << " ";}std::cout << std::endl;for (size_t i = 0; i < s2.size(); i++){std::cout << s2[i] << " ";}std::cout << std::endl;}void TestString2(){string s1;string s2("bit");std::cout << s1 << std::endl;std::cout << s2 << std::endl;s1.push_back('b');std::cout << s1 << std::endl;s2.push_back(' ');std::cout << s2 << std::endl;s1 += 'a';std::cout << s1 << std::endl;s2.insert(1, 'a');std::cout << s2 << std::endl;std::cout << s2.size() << std::endl;std::cout << s2.capacity() << std::endl;}void TestString3(){string s1;std::cin >> s1;std::cout << s1 << std::endl;}
}
测试代码:
#define _CRT_SECURE_NO_WARNINGS 1
#include"string.h"
int main()
{yyw::TestString1();yyw::string s3("hello");yyw::string::iterator it = s3.begin();while (it != s3.end()){std::cout << *it << " ";it++;}std::cout << std::endl;for (auto e : s3){std::cout << e << " ";}std::cout << std::endl;yyw::TestString2();yyw::TestString3();return 0;
}
相关文章:

【C++教程从0到1入门编程】第八篇:STL中string类的模拟实现
一、 string类的模拟实现 下面是一个列子 #include <iostream> namespace y {class string{public: //string() //无参构造函数// :_str(nullptr)//{}//string(char* str) //有参构造函数// :_str(str)//{}string():_str(new char[1]){_str[0] \0;}string(c…...

学生时期学习资源同步-1 第一学期结业考试题6
原创作者:田超凡(程序员田宝宝) 版权所有,引用请注明原作者,严禁复制转载...

迁移学习怎么用
如果想实现一个计算机视觉应用,而不想从零开始训练权重,比方从随机初始化开始训练,更快的方式是下载已经训练好权重的网络结构,把这个作为预训练,迁移到你感兴趣的新任务上。ImageNet、PASCAL等等数据库已经公开在线。…...

医疗手持智能终端读取条码二维码的难点有哪些?
在医疗科技行业信息化的大潮中,医疗手持式智能终端的应用越发普及,医疗手持式智能终端对条码二维码技术应用显得尤为关键,作为信息朔源载体的条码二维码读取方面,在实际应用中却面临着诸多问题,我们该如何应对…...

Python小设计
1. 五个PPT上的界面打印【print、input函数】 (1)英雄商城登陆界面 print(英雄联盟商城登录界面 ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~1. 用户登录2. 新用户注册3. 退出系统 ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~ * ~…...
今日讲讲父子传值~
今天来讲讲父子传值中的几种方法~ 项目中往往会把一些常用的公共代码抽离出来,写成一个子组件。或者在一个页面中的代码太多,可以根据功能的不同抽离出相关代码写成子组件,这样代码结构会更加简洁明了,后续维护更加方便。…...

三、HarmonyOS 应用开发入门之运行Hello World
目录 1、课程对象 1.1、有移动端开发经验 1.2、无移动端开发经验 1.3、对 HarmonyOS 感兴趣 2、DevEco Studio 的使用 2.1、DevEco Studio 的关键特性 智能代码编辑 低代码开发 多段双向实时预览 多端模拟仿真 2.2、安装配置 DevEco Studio 2.2.1、官网开发工具下载地…...

国科大网络行为学导论代码作业--更新中
一、Xray安装 参考自:Xray的安装与使用(超详细)_xray使用教程-CSDN博客 下载网址:Releases chaitin/xray GitHub 解压 双击安装 生成证书 cd到xray目录,生成证书 复制链接 然后cd到xray目录 .\xray_windows_amd6…...
JAVA后端开发面试基础知识(九)——SpringBoot
启动原理 SpringBoot启动非常简单,因其内置了Tomcat,所以只需要通过下面几种方式启动即可: @SpringBootApplication(scanBasePackages = {"cn.dark"}) public class SpringbootDemo {public static void main(String[] args) {// 第一种SpringApplication.run(S…...

C#调用Halcon出现尝试读取或写入受保护的内存,这通常指示其他内存已损坏。System.AccessViolationException
一、现象 在C#中调用Halcon,出现异常提示:尝试读取或写入受保护的内存,这通常指示其他内存已损坏。System.AccessViolationException 二、原因 多个线程同时访问Halcon中的某个公共变量,导致程序报错 三、测试 3.1 Halcon代码 其中tsp_width…...
ts基础知识
1. any 类型,unknown 类型,never 类型 TypeScript 有两个“顶层类型”(any和unknown),但是“底层类型”只有never唯一一个 1、any 1.1 基本含义 any 类型表示没有任何限制,该类型的变量可以赋予任意类型的…...
KLayout Python Script ------ 绘制1个 Box 物体
KLayout Python Script ------ 绘制两个 Box 物体 引言正文引言 本人通常使用 IPKISS 进行版图绘制,然而很多时候,IPKISS 的功能十分鸡肋。因此,萌生了一种自己写绘制软件的想法,因为 IPKISS 绘制的版图最终也是使用 KLayout 来呈现的,因此,再研究了 KLayout 提供的 API…...
c# 编辑、删除一条数据
1、编辑数据 [HttpPost] public MessageModel<string> Put([FromBody] Dtable request) { var data new MessageModel<string>(); request.UPDATETIME DateTime.Now; if (request.ID>0) { …...
高性能服务系列【八】C10M时代,网络IO库需要重建
在目前网络上能搜索到的,关于网络IO模型的文章,基本都是关于多路复用的iocp/epoll的,这些技术是为了解决C10K问题而提出的解决方案。现代网卡已经普遍支持10Gb,100Gb也不少见,这些解决方案已经无法提升性能的需求。 我…...
Go语言与Rust哪一个更有发展前景?
Go语言和Rust都是目前非常受欢迎的编程语言,它们各自具有独特的优势和适用场景。关于哪一个更有发展前景,这实际上取决于多个因素,包括个人偏好、项目需求、社区支持以及未来技术的发展趋势等。 Go语言是由Google推出的,具有简洁…...
STM32使用定时器驱动电机
STM32使用定时器驱动电机 1、对定时器进行初始化配置1.1、include "encoder.c"文件 主函数 1、对定时器进行初始化配置 1.1、include "encoder.c"文件 #include "encoder.h"void TIM4_Encoder_Init(u16 arr,u16 psc) { GPIO_InitTypeDef GPIO…...

C语言游戏实战(4):人生重开模拟器
前言: 人生重开模拟器是前段时间非常火的一个小游戏,接下来我们将一起学习使用c语言写一个简易版的人生重开模拟器。 网页版游戏: 人生重开模拟器 (ytecn.com) 1.实现一个简化版的人生重开模拟器 (1) 游戏开始的时…...

MVC架构模式学习笔记(动力节点老杜2022)
GitHub代码笔记:laodu-mvc: 动力节点学习javaweb中的mvc笔记。 文章目录 1.视频链接 2.不使用MVC架构模式程序存在的缺陷 3.MVC架构模式理论基础 4.JavaEE设计模式-DAO模式 5.pojo & bean & domain 6.业务层抽取以及业务类实现 7.控制层 8.MVC架构模式与三…...

docker常用操作-docker私有仓库的搭建(Harbor),并将本地镜像推送至远程仓库中。
1、docker-compose安装,下载docker-compose的最新版本 第一步:创建docker-compose空白存放文件vi /usr/local/bin/docker-compose 第二步:使用curl命令在线下载,并制定写入路径 curl -L "https://github.com/docker/compos…...
什么是MVC
MVC的全名是Model View Controller,是模型(model)-视图(view)-控制器(controller)的缩写,是一种软件设计典范。它是用一种业务逻辑、数据与界面显示分离的方法来组织代码,将众多的业务逻辑聚集到一个部件里面ÿ…...

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型
摘要 拍照搜题系统采用“三层管道(多模态 OCR → 语义检索 → 答案渲染)、两级检索(倒排 BM25 向量 HNSW)并以大语言模型兜底”的整体框架: 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后,分别用…...
谷歌浏览器插件
项目中有时候会用到插件 sync-cookie-extension1.0.0:开发环境同步测试 cookie 至 localhost,便于本地请求服务携带 cookie 参考地址:https://juejin.cn/post/7139354571712757767 里面有源码下载下来,加在到扩展即可使用FeHelp…...

【kafka】Golang实现分布式Masscan任务调度系统
要求: 输出两个程序,一个命令行程序(命令行参数用flag)和一个服务端程序。 命令行程序支持通过命令行参数配置下发IP或IP段、端口、扫描带宽,然后将消息推送到kafka里面。 服务端程序: 从kafka消费者接收…...
PHP和Node.js哪个更爽?
先说结论,rust完胜。 php:laravel,swoole,webman,最开始在苏宁的时候写了几年php,当时觉得php真的是世界上最好的语言,因为当初活在舒适圈里,不愿意跳出来,就好比当初活在…...
IGP(Interior Gateway Protocol,内部网关协议)
IGP(Interior Gateway Protocol,内部网关协议) 是一种用于在一个自治系统(AS)内部传递路由信息的路由协议,主要用于在一个组织或机构的内部网络中决定数据包的最佳路径。与用于自治系统之间通信的 EGP&…...

2021-03-15 iview一些问题
1.iview 在使用tree组件时,发现没有set类的方法,只有get,那么要改变tree值,只能遍历treeData,递归修改treeData的checked,发现无法更改,原因在于check模式下,子元素的勾选状态跟父节…...

屋顶变身“发电站” ,中天合创屋面分布式光伏发电项目顺利并网!
5月28日,中天合创屋面分布式光伏发电项目顺利并网发电,该项目位于内蒙古自治区鄂尔多斯市乌审旗,项目利用中天合创聚乙烯、聚丙烯仓库屋面作为场地建设光伏电站,总装机容量为9.96MWp。 项目投运后,每年可节约标煤3670…...

深入解析C++中的extern关键字:跨文件共享变量与函数的终极指南
🚀 C extern 关键字深度解析:跨文件编程的终极指南 📅 更新时间:2025年6月5日 🏷️ 标签:C | extern关键字 | 多文件编程 | 链接与声明 | 现代C 文章目录 前言🔥一、extern 是什么?&…...

ABAP设计模式之---“简单设计原则(Simple Design)”
“Simple Design”(简单设计)是软件开发中的一个重要理念,倡导以最简单的方式实现软件功能,以确保代码清晰易懂、易维护,并在项目需求变化时能够快速适应。 其核心目标是避免复杂和过度设计,遵循“让事情保…...
Pinocchio 库详解及其在足式机器人上的应用
Pinocchio 库详解及其在足式机器人上的应用 Pinocchio (Pinocchio is not only a nose) 是一个开源的 C 库,专门用于快速计算机器人模型的正向运动学、逆向运动学、雅可比矩阵、动力学和动力学导数。它主要关注效率和准确性,并提供了一个通用的框架&…...