C++初阶:适合新手的手撕string类(模拟实现string类)

上次讲了常用的接口:C++初阶:初识STL、String类接口详细讲解(万字解析)
今天就来进行模拟实现啦


文章目录

  • 1.基本结构与文件规划
  • 2.构造函数(constructor)
    • 2.1构造函数
      • 2.1.1无参有参分开
      • 2.1.2利用缺省参数合起来
    • 2.2拷贝构造
    • 2.3模拟c_str()函数
  • 3.析构函数(destructor)
  • 4.operator=
  • 5.迭代器(iterator)
    • 6.1size()与capacity()
    • 6.2 reserve()函数
  • 7. modify
    • 7.1push_back()、append()和operator+=
    • 7.2clear()和swap()
  • 8.access(operator[])
  • 9.npos
  • 10.find()函数
  • 11.insert()和erase()
  • 13.operator<<和operator>>


先看一下大概有哪些部分:

请添加图片描述

1.基本结构与文件规划

请添加图片描述

  • string.h头文件:包含类的全部(函数的声明与定义)
  • test.cpp源文件:进行调用test函数,测试和完善功能

基本结构:

namespace MyString
{class string{public://各种函数private:int _size;//有效字符的数量int _capacity;//开的空间大小char* _str;//没有设计成模版,就直接用char数组了};
}

2.构造函数(constructor)

2.1构造函数

2.1.1无参有参分开

		string()//空参的{_size = 0;_capacity = 0;_str = new char[1];_str[0] = '\0';}string(const char* str){_size = strlen(str);_capacity = _size;_str = new char[_size + 1];//加一是给 \0strcpy(_str, str);}

2.1.2利用缺省参数合起来

		string(const char* str="")//不是" "{_size = strlen(str);_capacity = _size;_str = new char[_size + 1];//加一是给 \0strcpy(_str, str);}

2.2拷贝构造

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

2.3模拟c_str()函数

目前还没有进行流插入和流提取的重构,只能利用这个来输出string里的内容

const char* c_str()const
{return _str;
}

利用test1函数来看是否正确:

相同的命名空间会进行合并的

namespace MyString
{void test1(){string s1;cout << s1.c_str() << endl;string s2("abc");cout << s2.c_str() << endl;string s3(s2);cout << s3.c_str() << endl;}
}int main()
{MyString::test1();return 0;
}

请添加图片描述


3.析构函数(destructor)

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

4.operator=

		string& operator=(const string& s)//创建好新的空间(复制跟赋值)。再清理旧空间指向新的{char* tmp = new char[s._capacity+1];//创建好新空间strcpy(tmp, s._str);_size = s._size;_capacity = s._capacity;//复制跟赋值delete[] _str;_str = tmp;//清理,指向新的return *this;}

5.迭代器(iterator)

		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;}

#6.capacity

6.1size()与capacity()

		size_t size()const{return _size;}size_t capacity()const{return _capacity;}

6.2 reserve()函数

		void reserve(size_t n){if (n > _capacity){_capacity = n;char* tmp = new char[_capacity + 1];strcpy(tmp, _str);delete[]_str;_str = tmp;//指向新的}}
namespace MyString
{void test2(){string s1;cout << s1.size() << endl;string s2("abc");cout << s2.size() << endl;string s3;s3 = s2;cout << s3.size() << endl;cout << s3.capacity() << endl;}
}int main()
{MyString::test2();return 0;
}

请添加图片描述


7. modify

7.1push_back()、append()和operator+=

		void push_back(char c){if (_size == _capacity){int newCapacity = _capacity == 0 ? 4 : 2 * _capacity;//有可能是空参构造reserve(newCapacity);}_str[_size++] = c;_str[_size] = '\0';}void append(const char* str){int len = strlen(str);if (len + _size > _capacity){reserve(len + _size);}strcpy(_str + _size, str);_size += len;}string& operator+=(char c){push_back(c);return *this;}string& operator+=(const char* str){append(str);return *this;}

这里其实大家也能看到,二者一个用于字符的添加,一个用于字符串的添加。

其实就可以一个写成另外一个的重载

7.2clear()和swap()

		void clear(){_size = 0;_str[0] = '\0';}void swap(string& s){std::swap(*this, s);//使用库里的swap}

对于swap,我们可以去使用库里面的,加上std::就行了

namespace MyString
{void test3(){string s1("abc");cout << s1.c_str() << endl;s1 += 'a';cout << s1.c_str() << endl;s1 += "xxx";cout << s1.c_str() << endl;}
}int main()
{MyString::test3();return 0;
}

请添加图片描述


8.access(operator[])

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

一个用于非常量,一个用于常量


9.npos

namespace MyString
{class string{public://各种函数private:int _size;//有效字符的数量int _capacity;//开的空间大小char* _str;//没有设计成模版,就直接用char数组了static size_t npos;};size_t string::npos = -1;
}

10.find()函数

		// 返回c在string中第一次出现的位置size_t find(char c, size_t pos = 0) const//默认从0开始找{for (int i = 0; i < _size; i++){if (_str[i] == c){return i;}}return npos;}// 返回子串s在string中第一次出现的位置size_t find(const char* s, size_t pos = 0) const{char* ret = strstr(_str, s);//返回第一次出现的位置if (ret == nullptr)//没找到{return npos;}return ret - _str ;}
namespace MyString
{void test4(){string s1("abc");cout << s1.find('a') << endl;cout << s1.find("bc");}
}int main()
{MyString::test4();return 0;
}

请添加图片描述


11.insert()和erase()

		// 在pos位置上插入字符c/字符串str,并返回该字符的位置string& insert(size_t pos, char c){if (_size = _capacity)//先看空间够不够{int newcapacity = _capacity == 0 ? 4 : 2 * _capacity;reserve(newcapacity);}//先向后移一格int end = _size;// \0也往后while (end >= (int)pos)//pos一般为size_t类型,此时,end会整型提升(有符号整数类型会被提升为无符号整数类型。){_str[end + 1] = _str[end];end--;}_str[pos] = c;_size++;return *this;}string& insert(size_t pos, const char* str){int len = strlen(str);if (_size +len>= _capacity){reserve(_size + len);}//先向后移len个格int end = _size;// \0也往后while (end >= (int)pos){_str[end + len] = _str[end];end--;}strncpy(_str + pos, str,len);_size += len;return *this;}// 删除pos位置上的元素,并返回该元素的下一个位置string& erase(size_t pos, size_t len= npos){if (len == npos || len + pos >= _size){_str[pos] = '\0';_size = pos;}else{int start = pos;while (_size - start - len+1 > 0){_str[start] = _str[start + len];start++;}}return *this;}

测试:

namespace MyString
{void test5(){string s1("abc");cout << s1.c_str() << endl;int pos = s1.find("bc");s1.insert(pos, "xxx");cout << s1.c_str() << endl;s1.erase(pos,2);cout << s1.c_str() << endl;}
}int main()
{MyString::test5();return 0;
}

请添加图片描述


#12.substr()函数

		string substr(size_t pos, size_t len = npos){int end = pos + len;//end作为结束位置if (len == npos || pos + len >= _size){end = _size;//当这两种情况,结束位置就是最后\0了}//开始pos到end赋值到新的一个空间,再返回string s;reserve(end - pos);for (int i = pos; i < end; i++){s += _str[i];}return s;}

13.operator<<和operator>>

	ostream& operator<<(ostream& out, string& s)//不需要放在类内,不需要是友元{for (auto e : s){out << e;}return out;}istream& operator>>(istream& in, string& s){s.clear();char ch = in.get();//使用cin拿不到空格和换行,二者其实还是字符,就用这个getwhile (ch != ' ' && ch != '\n'){s += ch;ch = in.get();}return in;}

测试:

namespace MyString
{void test6(){string s1;cin >> s1;cout << s1 << endl;int pos = s1.find('b');string s2 = s1.substr(pos, 2);cout << s2 << endl;}
}int main()
{MyString::test6();return 0;
}

请添加图片描述


好啦,今天就到这里啦,感谢大家支持!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/666207.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Pyecharts炫酷散点图构建指南【第50篇—python:炫酷散点图】

文章目录 Pyecharts炫酷散点图构建指南引言安装Pyecharts基础散点图自定义散点图样式渐变散点图动态散点图高级标注散点图多系列散点图3D散点图时间轴散点图笛卡尔坐标系下的极坐标系散点图 总结&#xff1a; Pyecharts炫酷散点图构建指南 引言 在数据可视化领域&#xff0c;…

[C++]:15.继承

继承 一.继承&#xff1a;1.继承的概念和基本操作&#xff1a;1.概念&#xff1a;2.基本操作&#xff1a; 2.继承格式和多种继承方法&#xff1a;1.基本继承格式&#xff1a;2.继承关系访问限定符 3.子类对象和父类对象之间的赋值&#xff1a;1.为什么存在赋值兼容转换&#xf…

第十二篇【传奇开心果系列】Python的OpenCV技术点案例示例:视频流处理

传奇开心果短博文系列 系列短博文目录Python的OpenCV技术点案例示例短博文系列短博文目录一、前言二、视频流处理介绍三、实时视频流处理示例代码四、视频流分析示例代码五、归纳总结系列短博文目录 Python的OpenCV技术点案例示例短博文系列 短博文目录 一、前言 OpenCV视频…

程序报错无法打开源文件stdafx.h

在运行代码时&#xff0c;代码中头文件突然报错程序无法打开源文件stdafx.h include “stdafx.h”,编译器就说无法打开源文件&#xff0c;直接上干货解决方法是&#xff1a; 1.打开项目 ->项目属性&#xff08;最后一个&#xff09;-> C/C ->常规&#xff0c; 2在附…

【工作周志】240129-240204

本周学习了AXI相关的内容 AMBA &#xff08;Advanced Microcontroller Bus Architecture&#xff09; AXI &#xff08;Advanced eXtensible Interface&#xff09; ARM公司提出&#xff0c;AMBA3.0协议中重要组成部分&#xff0c;是一种面向高性能、高带宽、低延迟的片内总线…

【c++】vector用法详解

vector用法详解 vector定义vector容器的构造函数vector容器内元素的访问1.通过下标 [ ]来访问2.通过迭代器来访问3.通过范围for来访问 vector常用函数的用法解析1.size()2.clear()3.capacity()4.reserve()5.resize()6.shrink_to_fit()7.pop_back()8.push_back()9.erase()10.in…

python基于django的公交线路查询系统mf383

1.个人信息的管理&#xff1a;对用户名&#xff0c;密码的增加、删除等 2.线路信息的管理&#xff1a;对线路的增加、修改、删除等 3.站点信息的管理&#xff1a;对站点的增加、修改、删除等 4.车次信息的管理&#xff1a;对车次的增加、修改、删除等 5.线路查询、站点查询 …

已解决: ModuleNotFoundError: No module named ‘tensorflow‘ 问题

博主猫头虎的技术世界 &#x1f31f; 欢迎来到猫头虎的博客 — 探索技术的无限可能&#xff01; 专栏链接&#xff1a; &#x1f517; 精选专栏&#xff1a; 《面试题大全》 — 面试准备的宝典&#xff01;《IDEA开发秘籍》 — 提升你的IDEA技能&#xff01;《100天精通鸿蒙》 …

nba2k24 球魁面补【23-24通用】

nba2k24 球魁面补 nba2k23-nba2k24通用 球魁面补 下载地址&#xff1a; https://www.changyouzuhao.cn/9841.html

蓝桥杯省赛无忧 课件91 高斯消元

01 算法概述 02 问题引入 03 算法分析 04 例题

Linux 多线程 | 线程的概念

线程的概念 线程是一个执行分支&#xff0c;执行粒度比进程更细&#xff0c;调度成本更低&#xff1b; 线程是进程内部的一个执行流&#xff1b; 线程是CPU调度的基本单位&#xff0c;进程是承担分配系统资源的基本实体。 之前我们学习过虚拟地址空间的知识&#xff0c;知道…

NetSuite 权限不足用户如何查询完整数据

假设我们做了一个Saved Search&#xff0c;用于统计所有涉及库存的事务类型&#xff0c;包括出入库、库存调整、生产报工、拆解、Standalone Invoice和Bill&#xff0c;等等。通过合计这些事务类型&#xff0c;我们就可以得到一个存货报表&#xff0c;能够得到任一时间点的库存…

线程同步解析

一 线程同步 1 同步的意义 现实中抢票可能没票了还在抢票&#xff0c;然后线程就会一直在加锁解锁&#xff0c;就会导致其它线程抢不到锁而产生饥饿问题&#xff0c;我们前面也提过usleep就是让线程被切换&#xff0c;能让其它线程去申请锁&#xff0c;这种方式并不好&#xf…

蓝桥杯备战——13.PCF8591芯片的使用

目录 1.芯片简介2.读写时序3.控制字4.代码封装库5.原理图分析6.使用示例 1.芯片简介 截取自NXP的PCF8591芯片数据手册&#xff0c;我把重点关注部分划出来了&#xff0c;请务必自行阅读一遍数据手册&#xff01; 2.读写时序 ①器件地址&#xff1a; Bit0决定是读还是写操作&…

最新GPT4.0使用教程,AI绘画,GPT语音对话使用,DALL-E3文生图

一、前言 ChatGPT3.5、GPT4.0、GPT语音对话、Midjourney绘画&#xff0c;文档对话总结DALL-E3文生图&#xff0c;相信对大家应该不感到陌生吧&#xff1f;简单来说&#xff0c;GPT-4技术比之前的GPT-3.5相对来说更加智能&#xff0c;会根据用户的要求生成多种内容甚至也可以和…

优质成长:新生儿补充维生素B6的关键注意事项

引言&#xff1a; 维生素B6&#xff0c;作为B族维生素的一员&#xff0c;对于新生儿的神经系统发育和代谢功能至关重要。本文将深入探讨维生素B6的作用、新生儿补充的必要性&#xff0c;以及在补充维生素B6时应该注意的事项&#xff0c;为父母提供科学、全面的育儿指南。 第一…

WebChat——一个开源的聊天应用

Web Chat 是开源的聊天系统&#xff0c;支持一键免费部署私人Chat网页的应用程序。 目录树 TOC &#x1f44b;&#x1f3fb; 开始使用 & 交流&#x1f6f3; 开箱即用 A 使用 Docker 部署B 使用 Docker-compose 部署C 使用 Jar包 本地部署 ⌨️ 本地开发&#x1f91d; 参与…

开源浏览器Firefox:使用Docker本地部署并远程访问进行测试

&#x1f308;个人主页&#xff1a;聆风吟 &#x1f525;系列专栏&#xff1a;网络奇遇记、数据结构 &#x1f516;少年有梦不应止于心动&#xff0c;更要付诸行动。 文章目录 &#x1f4cb;前言一. 部署Firefox二. 本地访问Firefox三. Linux安装Cpolar四. 配置Firefox公网地址…

网络原理-TCP/IP(5)

TCP协议 延迟应答 它也是基于滑动窗口,提高效率的一种机制,结合滑动窗口以及流量控制,能够以延迟应答ACK的方式,把反馈的窗口,搞大.核心在于允许范围内,让窗口尽可能大. 如果接收数据的主机立刻返回ACK应答,这时候返回的窗口可能比较小. 1.假设接收端缓冲区为1M.一次收到了5…

树状数组相关

前置细节 &#xff0c;得到转二进制后&#xff0c;从右往左数第一个1与之前所有的0构成数树状数组最终形成如上图结构维护的是的信息&#xff0c;如即到上层&#xff0c;包含当前区间的大区间&#xff0c;如即到同一层的前一个&#xff0c;与当前区间无关的另一同等大小的区间…