模拟实现string

news2025/7/9 16:56:25

image-20221116120149781

第一部分:构造,析构,拷贝构造,赋值重载,打印函数这几个大头写出来先

string类框架


namespace xxx
{
class string
{
public:
//
//

private:
char* _str;
size_t _size;
size_t _capacity;
   const static size_t npos = -1;//c++允许const static整数可以定义在类里-但只有几个特定的整数可以这样
};
}

构造函数

	//构造函数
		string(const char* str = "")//给缺省值
        //  :_size(strlen(str))
		//	,_capacity(strlen(str))
		//这里不用初始化列表,这里要一个个strlen赋值比较麻烦
        {
			_size = strlen(str);//_size为有效字符的长度
			_capacity = _size;//capacity初始化都为有效字符的长度
			_str = new char[_capacity + 1];//留一个字符给'\0'
			strcpy(_str, str);//拷贝过去
		}

析构函数

	//析构函数
		~string()
		{
			delete[] _str;
			_str = nullptr;
			_size = _capacity = 0;
		}

拷贝构造

		//拷贝构造-深拷贝噢
		string(const string& s)//传引用
		{
			_str = new char[s._capacity + 1];//留一个字符给斜杠0
			strcpy(_str, s._str);//_str拷贝
			_size = s._capacity;
			_capacity = s._capacity;
		}

赋值重载

	//赋值重载
		string& operator=(const string& s)//传引用
		{
			if (this != &s)
			{
				char* tmp = new char[s._capacity + 1];//开新空间
				strcpy(tmp, s._str);//拷贝
				delete[]_str;//删除_str的就空间
				_str = tmp;//指向新空间
				_size = s._size;
				_capacity = s._capacity;
			}
			return *this;
		}

对于**【为什么要开一块新空间然后还要把之前的旧空间释放掉**】有疑惑的朋友可以来看这篇https://blog.csdn.net/m0_71841506/article/details/127291467

通过c_str打印

	//返回一个指向正规C字符串的指针常量, 内容与本string串相同
		const char* c_str()const
		{
		return 	_str;
		}

size和capacity

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

基本上写到这就能输出hello world拉!

image-20221114203417918

第二部分:各类功能和升级

迭代器iterator

	typedef char* iterator;
		iterator begin()
		{
			return _str;//返回第一个指针位置
		}
		iterator end()
		{
			return _str + _size;//返回最后一个有效字符的下一个位置
		}

实现iterator之后我们可以这样玩

image-20221114204504544

还能用范围for

image-20221114204913418

但是我们把iterator实现的部分注释掉的话,范围for就用不了了;又或者说把iterator begin()改为iterator Begin()我们会发现iterator打印还能用,但范围for也用不了了;这说明范围for底层与iterator实现有关!至于为啥嘛。。。知道的可以在评论区留言~

方括号重载

char &operator[](int pos)const//方括号重载
		{
			return _str[pos];
		}

重载之后我们就能用数组【】访问

image-20221114214055011

push_back追加字符

		void push_back(char ch)//尾插字符
		{
			if (_size == _capacity)//扩容
			{
				size_t newcapacity = _capacity == 0 ? 4 : 2 * _capacity;
				reserve(newcapacity);
			}
			_str[_size] = ch;//在末尾追加字符ch
			_size++;
			_str[_size] = '\0';//加上\0
	}

operator+=追加字符运算符重载

	string& operator+=(char ch)//追加字符重载
		{
			push_back(ch);
			return *this;
		}

append追加字符串

	//后面追加追加字符串
		void append(const char* str)
		{
			size_t len = strlen(str);//记录str有效字符串长度
			if (_size + len > _capacity)//直接扩容
			{
				reserve(_size + len);
		}
			strcpy(_str + _size, str);//strcpy把str连同\0一起拷贝过来了,所以不需要考虑\0
			_size += len;	
		}

operator+=追加字符串运算符重载

	string& operator+=(const char* str)//追加字符串重载
		{
			append(str);
			return *this;
		}

clear清除字符串内有效字符

	void clear()//清除所有有效字符
		{
			_str[_size] = '\0';
			_size = 0;
		}

流插入<<

	ostream& operator<<(ostream& out, const string&s )//流插入
	{
		for (size_t i = 0; i < s.size(); ++i)
		{
			out << s[i];
			
		}
		return out;
	}

流提取>>

流提取是当遇到空格或者换行则截断提取,那么我们要按照这个底层来实现它


	istream& operator>>(istream& in, string& s)//流提取
	{
		s.clear();//如果流提取之前s里面有字符,则要清除
		char buff[128] = { '\0' };//开一块空间做中转站
		size_t i = 0;
		char ch= in.get();//用ch一个个接收s的字符
		while (ch != ' ' && ch != '\n')//当ch接收到换行或者空格则停止
		{
			if (i == 127)
			{
				//满了-buff尾接给s
				s += buff;
				i = 0;
			}
			//没满ch继续接收
			buff[i++] = ch;
			ch = in.get();
		}
		if (i>0)//如果没满则buff没有尾插到s后面;又或者buff满过的部分尾接给了s,但剩下的部分没有尾接给s
		{
			buff[i] = '\0';//要携带\0 !
			s += buff;
}
		return in;
	}

那么我们就能用流插入打印,流提取我们输入的内容拉

image-20221114223428759

image-20221115091827032

insert

size_t和int类型同时在操作符两边,会造成整形提升-小的往大的提升(int->size_t);那么int也转变为无符号数,则没有负数,比较大小会出错,会进入死循环;

insert插入字符

在pos位置插入字符ch

string& insert(size_t pos, char ch)//插入字符-
		{
			assert(pos <=_size);

			if (_size == _capacity)//扩容
			{
				size_t newcapacity = _capacity == 0 ? 4 : 2 * _capacity;
				reserve(newcapacity);
			}
			//挪动数据
			//size_t end = _size+1;//(size_t)无符号版
			//while (end>pos)
			//{
			//	_str[end] = _str[end-1];
			//	end--;
			//}
			//(int)有符号版
			int end = _size-1;
			while (end >=(int) pos)
			{
				_str[end + 1] = _str[end];
				end--;
			}
			//插入字符
			_str[pos] = ch;
			_size++;
			_str[_size] = '\0';
			return *this;
		}

image-20221115101414713

insert插入字符串

在pos位置插入len长的字符串

string& insert(size_t pos, const char* str)//插入字符串
		{
			//扩容
			size_t len = strlen(str);//记录要插入字符串的长度
			if (_size + len > _capacity)
			{
				reserve(_size + len);
			}
			
			//挪动数据
			//(size_t)无符号版
			//size_t end = _size+1;//+1把\0也挪到_str[_szie+len-1]的位置上
			//while (end > pos)
			//{
			//	_str[end + len-1] = _str[end-1];
			//	end--;
			//}
			//有(int)符号版
			int end = _size;//把\0也挪到_str[_size+len]位置上
			while (end >= (int)pos)
			{
				_str[end + len] = _str[end];
				--end;
			}
			//插入字符串
			strncpy(_str + pos, str, len);
			_size += len;
			return *this;
		}

image-20221115104633841

erase 删除

从pos位置往后删除len个字符,不给len则按缺省值npos删除(全删完)

string& erase(size_t pos, size_t len = npos)
		{
			assert(pos < _size);

			if (len == npos || len >= _size - pos)
                //如果len长度大于pos之后的有效字符串长度,那么也在pos位置\0
            {
				_str[pos] = '\0';
				_size = pos;
			}
			else
			{
				strcpy(_str + pos, _str + pos + len);
				_size -= len;
			}

			return *this;
		}

image-20221115114423146

find

find寻找字符

	size_t find(char ch, size_t pos=0)//寻找字符-给缺省值从0开始找
		{
			assert(pos < _size);
			while (pos<_size)
			{
				if (_str[pos] == ch)
				{
					return pos;//,返回下标
				}
					++pos;
			}
			return npos;//没找到返回npos
		}

image-20221115120002408

find寻找字符串

	size_t find(const char* s, size_t pos = 0)//寻找字符串-给缺省值从0开始找
		{
			assert(pos < _size);
			const char* ptr = strstr(_str, s);//strstr在字符串里面寻找字串-找到返回指针-没找到返回空
			if (ptr == nullptr)
			{
				return npos;//没找到
			}
			else
			{
				return ptr - _str;//指针相减得指针之间字符串数量-即为pos的位置
			}
		}

image-20221115120051552

高级版本的拷贝构造和赋值重载

拷贝构造

我们要用s1来拷贝构造s2,那么我们可以先用s1来构造tmp;然后在让指向tmp的指针和指向s2的指针交换指向;那么s2也完成了对s1内容的深拷贝;(这里要注意:要给原本是野指针的s2一个初始化空指针,交换后保证后续tmp析构时不越界析构其他空间【析构函数遇到空指针不析构】

image-20221116111951347

image-20221116112611005

	//拷贝构造-现代写法
		//s2(s1)
		string(const string& s)
			:_str(nullptr)//要初始化给个空指针,不然是野指针,tmp析构的时候会越界空间析构报错
			,_size(0)
			,_capacity(0)
		{
			string tmp(s._str);//用s构造tmp
			swap(_str,tmp._str);
			swap(_size, tmp._size);
			swap(_capacity, tmp._capacity);
		}

image-20221116112757177

但是我们盯着上面的现代写法,一共是交换了_str, _szie, _capacity一共三次深拷贝。那么能不能再简化呢?

我们通过查表格知道std库里的swap是先把对象实例化再交换;那么在这里是要进行三次深拷贝,深拷贝要开空间然后赋值。。。可见这样做代价非常大!

image-20221116113352782

string库里的swap是直接把两个string的指针交换这样代价相对小拉!

image-20221116113609167

所以我们可以写一个swap函数

string里的swap

void swap(string& s)
		{
		    std::swap(_str, s._str);
			std::swap(_size, s._size);
			std::swap(_capacity, s._capacity);
		}

那么简化后的版本是这样的:

//拷贝构造-现代写法-简化后
		//s2(s1)
		string(const string& s)
			:_str(nullptr)//要初始化给个空指针,不然是野指针,tmp析构的时候会越界空间析构报错
			,_size(0)
			,_capacity(0)
		{
			string tmp(s._str);//用s构造tmp
			//this->swap(tmp);this可以不写
			swap(tmp);
		}

赋值重载

//赋值重载-现代写法
		string& operator=( const string& s)//传引用
		{
			if (this != &s)
			{
				string tmp(s);//s构造tmp
				swap(tmp);//交换this和tmp指针
			}
			return *this;
		}

简化后版本

string& operator=(string& s)//传引用
		{
			if (this != &s)
			{
				swap(s);//不需要tmp做中间人直接交换
			
			}
			return *this;
		}

关于string类的模拟实现的总结就到这里拉,看到这里的观众老爷们不妨点赞收藏起来看吧~~~

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

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

相关文章

计算机视觉——python在一张图中画多条ROC线

在验证分类算法的好坏时&#xff0c;经常需要用到AUC曲线&#xff0c;而在做不同分类模型的对比实验时&#xff0c;需要将不同模型的AUC曲线绘制到一张图里。 计算机视觉——python在一张图中绘制多个模型的对比ROC线1. 小型分类模型对比&#xff0c;可以直接调用的2. 大型的CN…

七夕,程序员教你5个表白代码,2分钟学会,牢牢主抓她的心

七夕。一个有人欢喜有人愁的节日&#xff0c;虽然对一些单身人士不太友好&#xff0c;但还有不少人都在等这个节日进行表白。毕竟这个日子的成功率会高一些。 情人节少不了送花送礼物&#xff0c;作为一个程序员&#xff0c;当然不会在送什么礼物上给你指点一二&#xff0c;但…

适合骑车时候戴的耳机怎么选,列举五款在骑行佩戴的耳机推荐

相信大多数人在运动的过程中都会感觉到枯燥无力的感觉&#xff0c;为此也一直在寻找一些能够让我们在运动中保持最初的热诚&#xff0c;在最终的选择上&#xff0c;绝大多数都是选择了耳机&#xff0c;因为耳机能够产生美妙的音乐&#xff0c;将我们运动的枯燥做进一步的抵消&a…

【附源码】Python计算机毕业设计网络教育平台设计

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

【HMS Core】定位服务无法获取街道信息问题

问题描述&#xff1a; 华为HMS-定位服务无法获取省市街道信息 背景&#xff1a;在手机上集成华为定位服务的功能运行后&#xff0c;只能获取到经纬度&#xff0c;无法得到具体地址(城市/街道/建筑)。 配置环境&#xff1a;&#xff08;1&#xff09;手机型号&#xff1a;Red…

c语言实现通讯录(用三种方法来实现一个属于你的通讯录)

前沿&#xff1a; &#xff5e;&#xff5e;在一个周前&#xff0c;我用c语言实现了一个银行账号的功能&#xff0c;在总结当中我说了要实现一个通讯录&#xff0c;来实现人员的增删插改功能&#xff0c;而现在就是我实现它的时候&#xff01;&#xff01;&#xff0c;本文呢小…

基于python命令流及代码的Plaxis自动化建模

有限单元法在岩土工程问题中应用非常广泛&#xff0c;很多商业软件如Plaxis/Abaqus/Comsol等都采用有限单元解法。在使用各大软件进行数值模拟建模的过程中&#xff0c;您是否发现GUI界面中重复性的点击输入工作太繁琐&#xff1f;从而拖慢了设计或方案必选进程&#xff1f; 本…

SpringBoot八种bean的加载方式一学就会

目录 文章目录[toc]第一种bean的加载方式-配置文件第二种加载bean方式-注解和扫描创建第三方的bean对象第三种加载bean方式-不使用配置文件扩展-bean的加载方式扩展FactoryBean<>扩展-ImportResource导入配置文件扩展-proxyBeanMethods属性-产生代理对象第四种加载bean方…

一文读懂,python实现常用的数据编码和对称加密

相信很多使用 python 的小伙伴在工作中都遇到过&#xff0c;对数据进行相关编码或加密的需求&#xff0c;今天这篇文章主要给大家介绍对于一些常用的数据编码和数据加密的方式&#xff0c;如何使用 python 去实现。话不多说&#xff0c;接下来直接进入主题&#xff1a; 1、bas…

Windows OpenGL ES 图像灰度图

目录 一.OpenGL ES 图像灰度图 1.原始图片2.效果演示 二.OpenGL ES 图像灰度图源码下载三.猜你喜欢 零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 基础 零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 特效 零基础 Open…

Redis添加至windows服务

1.进入redis文件夹 打开redis所在文件夹&#xff0c;在资源管理器地址栏输入cmd&#xff0c;打开管理员命令提示符窗口。 2.配置服务参数 打开redis.windows.conf文件&#xff0c;找到“requirepass”配置项&#xff0c;此处设置redis密码&#xff0c;如果不需要密码&#xff0…

Python如何使用HanNLP工具

Python如何使用HanNLP工具目录系统安装配置 JDK 1.8系统安装 Visual C 2015下载 HanNLP 包测试HanNLP工具目录 目标&#xff1a;使用pycharm调用HanNLP工具完成对文本的分词、自动摘要、关键词提取等任务。 系统安装配置 JDK 1.8 1、windows环境下载 JDK 1.8 2、安装 JDK 1.…

Oracle PrimaveraUnifier空间管理器(Space Manager)

目录 前言 介绍 前言 在Oracle Primavera Unifier设施和资产管理基础产品中除了业务流程及配置管理器之外&#xff0c;其预配置设计还包含本文要介绍的空间管理器&#xff0c;即Space Manager 在Unifier中&#xff0c;空间管理器是用户可以执行设施管理任务的地方。空间管…

xv6---Lab1: Xv6 and Unix utilities

目录 参考资料&#xff1a; 1.1进程和内存 1.2 I/O 和文件描述符 1.3管道 源码&#xff1a; 调试环境搭建 sleep PingPong primes find xargs 参考资料&#xff1a; Lab: Xv6 and Unix utilities xv6-book翻译(自用&#xff09;第一章 - 知乎 1.1进程和内存 一个xv…

html中css的基础学习

小李胖了吗 I 都说秋天适合思念&#xff0c;其实更适合见面【小李胖了吗 I 都说秋天适合思念&#xff0c;其实更适合见面】 https://www.bilibili.com/video/BV19g411B7uL/?share_sourcecopy_web&vd_source385ba0043075be7c24c4aeb4aaa73352 通过本博文的学习&#xff0c…

常见的软件测试面试题,千万别答错了

软件测试的童鞋们&#xff0c;在面试测试工作时&#xff0c;一定遇到面试官问过这个问题&#xff1a; 软件测试的目的意义是什么&#xff1f;大家是怎么回答的呢&#xff1f;如果这个问题回答好了&#xff0c;说明你对软件测试工作的价值与意义了如指掌。 有经验的测试人员可…

[数据结构]栈和队列面试题解析

作者&#xff1a; 华丞臧. 专栏&#xff1a;【数据结构】 各位读者老爷如果觉得博主写的不错&#xff0c;请诸位多多支持(点赞收藏关注)。如果有错误的地方&#xff0c;欢迎在评论区指出。 推荐一款刷题网站 &#x1f449; LeetCode刷题网站 文章目录一、有效括号题目描述解题思…

freemarker+yml介绍 以及freemarker与JSP的区别

目录 1. freemarker介绍 2. freemarker使用步骤 2.1 在pom.xml引入freeMarker的依赖包 2.2 在springboot中添加freemarker配置 2.3 编写模板文件*.ftl(当做jsp使用即可) ​编辑 2.4 访问控制器后进行页面跳转 3. freemarker常用语法 3.1 取值 3.2 条件 3.3 循环 3…

超赞:不愧是“阿里内部Redis学习笔记”从头到尾,全是精华

近几年&#xff0c;随着移动互联网的飞速发展&#xff0c;我们享受着整个社会的技术进步带来的便利&#xff0c;但同时也给从业者带来了如何保证项目的高并发、低延时的技术挑战&#xff0c;相应的互联网技术也随之发生了重大变革&#xff0c;NoSQL技术得到了蓬勃的发展。 Red…

【JavaSE】抽象类与接口

文章目录抽象类的概念抽象类的语法抽象类的特性接口的概念接口的语法接口的特性多接口的实现接口的继承抽象类的概念 什么是抽象类呢&#xff1f;我们先来看一个例子&#xff1a;一个父类是动物类&#xff0c;两个子类一个狗狗类&#xff0c;一个猫猫类。 Animal中有一个dark…