【string类的简单模拟实现】

news2024/12/23 14:05:24

目录

1 类中成员变量的声明

2 迭代器

3 一些常用接口

4 六大默认函数

4.1 构造

4.2 拷贝构造

4.3 赋值运算符重载

4.4 析构

5 开空间&&增删查改

6 其他运算符重载


1 类中成员变量的声明

通过上一篇文章对string类的简单使用相信大家对于string类中成员变量已经很熟悉了,这里就不再多说了:

class string
	{
	private:
		char* _str;
		int _size;//当前有效数据个数
		int _capacity;//存储有效数据的最大容量,不包括'\0'
		static const size_t npos;
		typedef char* iterator;
		typedef const char* const_iterator;
    };

但是为了测试时避免与库里面的冲突我们就把该类放在一个独立的命名空间中:

namespace grm
{
	class string
	{
	private:
		char* _str;
		int _size;//当前有效数据个数
		int _capacity;//存储有效数据的最大容量,不包括'\0'
		static const size_t npos;
		typedef char* iterator;
		typedef const char* const_iterator;
    };
}

2 迭代器

        iterator begin()
		{
			return _str;
		}
		iterator end()
		{
			return _str + _size;
		}

		const_iterator begin()const
		{
			return _str;
		}
		const_iterator end()const
		{
			return _str + _size;
		}

3 一些常用接口

        size_t size()const     //无论是不是const对象都能够调用
		{
			return _size;
		}
		
		char* c_str()const
		{
			return _str;
		}
        
        char& operator[](size_t pos)
		{
			assert(pos < _size);
			return _str[pos];
		}

		const char& operator[](size_t pos)const
		{
			assert(pos < _size);
			return _str[pos];
		}
        void clear()const
		{
			_str[0] = '\0';
			_size = 0;
		}

		size_t capacity()const
		{
			return _capacity;
		}
        
        bool empty()const
		{
			return _size == 0;
		}

4 六大默认函数

4.1 构造

       string(const char* s)
			:_size(strlen(s))
			,_capacity(_size)
		{
			_str = new char[_capacity + 1];//多开出一个空间用来存储'\0'
			strcpy(_str, s);
		}

但是我们发现如果是无参又该怎么办?

有人会说像这样特殊处理一下就好了:

        string()
			:_str(new char[1])
			,_size(0)
			,_capacity(0)
		{
			_str[0] = '\0';
		}

这样处理是没有问题的,但是我们学过缺省参数,在这里能不能够给一个缺省值呢?缺省值又该给那个呢?

C++中是这样处理的:

        string(const char* s="")//给一个空字符串
			:_size(strlen(s))
			,_capacity(_size)
		{
			_str = new char[_capacity + 1];//多开出一个空间用来存储'\0'
			strcpy(_str, s);
		}

缺省值给的是一个空字符串。

4.2 拷贝构造

传统写法:

        //传统写法
		string(const string& s)
			:_size(s._size)
			,_capacity(s._capacity)
		{
			_str = new char[_capacity+1];
			strcpy(_str, s._str);
		}

这种写法就是自己手动开空间拷贝。

现代写法:我们发现上面我们已经实现好了默认的构造函数,我们可以直接用构造函数来解决

        void swap(string& s)
		{
			std::swap(s._str, _str);
			std::swap(s._capacity, _capacity);
			std::swap(s._size, _size);
		}
		//现代写法
		string(const string& s)
			:_str(nullptr)//必须要给初始值,否则交换后调用析构函数就会析构随机值会崩溃
			,_size(0)
			,_capacity(0)
		{
			string tmp(s._str);
			swap(tmp);
		}

其中需要注意的是使用这种方式用初始化列表将_str初始化成nullptr,不然交换后调用tmp的析构函数就会释放随机地址而出错。

4.3 赋值运算符重载

与拷贝构造一样,赋值运算符重载也分为传统版本和现代版本。

传统版本:

        //传统写法
		string& operator=(const string& s)
		{
			if (this != &s)
			{
				//这种方式如果new失败了抛异常就会有问题,我们并不想_str维护的空间被释放
				delete[] _str;
				_str = new char[s._capacity+1];
				strcpy(_str, s._str);
				_size = s._size;
				_capacity = s._capacity;
			}
			return *this;
		}

大家可以从注释中看见传统写法都是要自己手动开空间拷贝,而且大家也发现了代码中还多了一个判断this是否与&s相等,否则当直接delete_str后由于s与_str指向的是同一块空间,再对s解引用就是典型的野指针问题了。而且还有一个问题就是当我们new空间失败时我们并不想_str被释放,所以可以用下面这种写法:

        //传统写法
		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;
		}

而且大家发现没有这样写即使不用写上面的if判断程序依旧可以正常运行,但是加上判断当自己给自己赋值时可以减少拷贝。

现代写法 方法1:

        //现代写法 方法1:
		string& operator=(const string& s)
		{
			if (this != &s) //这里加上判断条件在自己给自己赋值能够减少拷贝,不加也是没有问题的
			{
				string tmp(s);
				swap(tmp);
			}
			return *this;
		}

现代写法 方法2:

        //现代写法 方法2:
		string& operator=(string s)
		{
			swap(s);
			return *this;
		}

这种方式就更加巧妙了,直接用了传值传参,而传值传参就是一个拷贝构造。

4.4 析构

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

5 开空间&&增删查改


		void reserve(size_t n) //这里要求开n个有效空间,不包括'\0'
		{
			if (_capacity < n)
			{
				char* tmp = new char[n + 1];
				strcpy(tmp, _str);
				delete[] _str;
				_str = tmp;
				_capacity = n;
			}
		}

		void resize(size_t n, char ch = '\0')
		{
			if (n > _size)
			{
				if (n > _capacity)
				{
					reserve(n);
				}
				memset(_str + _size, ch, n - _size);
			}
			_size = n;
			_str[_size] = '\0';
		}
        void push_back(char ch)
		{
			if (_capacity == _size)
				reserve(_capacity == 0 ? 4 : _capacity * 2);
			_str[_size] = ch;
			++_size;
			_str[_size] = '\0';
		}

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

		string& operator+=(char ch)
		{
			push_back(ch);
			return *this;
		}
		string& operator+= (const char* s)
		{
			append(s);
			return *this;
		}

		size_t find(char ch)
		{
			for (size_t i = 0; i < _size; i++)
				if (_str[i] == ch)
					return i;
			return -1;
		}

		size_t find(const char* s, size_t pos=0)
		{
			int i = pos, j = 0;//i是主串遍历,j是子串遍历
			while (i < _size && j < strlen(s))
			{
				if (s[j] == _str[i])
				{
					i++;
					j++;
				}
				else
				{
 					i = i - j + 1;
					j = 0;
				}
			}
			if (j == strlen(s))
				return i - j;
			else
				return -1;

		}
        string& insert(size_t pos, const char* s)//在pos位置插入,字符串从pos位置开始插
		{
			assert(pos <= _size);
			int len = strlen(s);
			if (_size + len > _capacity)
				reserve(_size + len);

			int end = _size + len - 1;
			int gap = _size - pos + 1;
			while (gap--)
			{
				_str[end] = _str[end - len];
				end--;
			}
			strncpy(_str + pos, s, len);
			_str[_size + len] = '\0';
			return *this;
		}

		string& erase(size_t pos, size_t len = npos)
		{
			assert(pos < _size);
			if (len == npos || pos + len >= _size)
			{
				_str[pos] = '\0';
				_size = pos+1;
			}

			strcpy(_str + pos, _str + len + pos);
			_size -= len;
			return *this;
		}

这些代码我们之前上顺序表都已经比较详细的介绍了,大家可以参考参考。

我们可以自己测试一下来看看:

 

 可以发现是没有多大问题的。


6 其他运算符重载

        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)const
		{
			return strcmp(_str, s._str) <= 0;
		}
		bool operator>=(const string& s)const
		{
			return strcmp(_str, s._str) >= 0;
		}bool operator!=(const string& s)const
		{
			return strcmp(_str, s._str) != 0;
		}
	

	ostream& operator<<(ostream& out, const string& s)
	{
		//out << s.c_str();//能这么写吗? 不能,也许我们会在字符中间插入一个'\0'

		for (auto& e : s)
		{
			out << e;
		}

		/*for (int i = 0; i < s.size(); i++)
		{
			out << s[i];
		}*/
		return out;
	}

	istream& operator>>(istream& in, string& s)
	{
		s.clear();//要加上这个,否则可能出错
		char ch = in.get();
		while (ch != '\0' && ch != '\n')
		{
			s += ch;
			ch = in.get();
		}
		return in;
	}

上面这几个比较可以自己重载成成员方法,当然也可以重载成全局函数。下面流提取运算符和流插入运算符我们在之前已经讲过了,里面需要注意的细节代码中都有注释。


有需要源码的老铁可以去博主的码云中获得:

nijhttps://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72https://gitee.com/monday-sky/text_cpp/commit/e4716c73bba49dd2bad8c86986d0cfe26beece72

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

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

相关文章

Android so库开发——Swig工具使用(五)

SWIG 是一种软件开发工具&#xff0c;它将 C 和 C 编写的程序与各种高级编程语言连接起来。这里我们用它来将 C/C 转换成 Java。 一、Swig安装 1、下载 官网&#xff1a;SWIG官网下载 源码链接 GitHub&#xff1a;https://github.com/swig/swig.git 这两个地址可能会出现无…

关于Java中的BigDecimal

文章目录为什么用BigDecimalBigDecimal构造方法通过静态方法创建BigDecimal对象BigDecimal常用API关于除法运算的roundingMode将BigDecimal转换为基本类型代码展示小结其他文章为什么用BigDecimal 使用float、double及其对应的包装类时&#xff0c;运算精度可能不满足需求 flo…

scrollTo() 无效问题处理

需求&#xff1a; 实现访问当前页面直接滚动到最底部 方案&#xff1a;window对象的scrollTo()方法 API介绍&#xff1a; 参数接收一个点&#xff08;文档坐标&#xff09;&#xff0c;让该点位于左上角。 可选参数为behavior(设置滚动的效果&#xff09; 错误案例&#xff1a;…

【My Electronic Notes系列——数制与编码】

目录 序言&#xff1a; &#x1f3c6;&#x1f3c6;人生在世&#xff0c;成功并非易事&#xff0c;他需要破茧而出的决心&#xff0c;他需要永不放弃的信念&#xff0c;他需要水滴石穿的坚持&#xff0c;他需要自强不息的勇气&#xff0c;他需要无畏无惧的凛然。要想成功&…

剑指 Offer II 005单词长度的最大乘积

给定一个字符串数组 words&#xff0c;请计算当两个字符串 words[i] 和 words[j] 不包含相同字符时&#xff0c;它们长度的乘积的最大值。假设字符串中只包含英语的小写字母。如果没有不包含相同字符的一对字符串&#xff0c;返回 0。 示例 1: 输入: words ["abcw"…

前端CSS第二阶段-001

&#x1f60a;博主页面&#xff1a;鱿年年 &#x1f449;博主推荐专栏&#xff1a;《WEB前端》&#x1f448; ​&#x1f493;博主格言&#xff1a;追风赶月莫停留&#xff0c;平芜尽处是春山❤️ 目录 第二阶段学习目标 一、Emmet语法 1.快速生成HTML结构语法 2.快速生成…

在Runtime下,IL2CPP与Mono打包对应的PSS内存占用问题

1&#xff09;在Runtime下&#xff0c;IL2CPP与Mono打包对应的PSS内存占用问题 ​2&#xff09;获得AssetBundle内部依赖关系的方法 3&#xff09;Unity 2019 Streaming Mipmap在某些情况下采样等级错误 4&#xff09;根据RenderDoc的数据&#xff0c;计算渲染量 这是第322篇UW…

2024年部分MBA/MEM项目提面日程已经开启,气氛已然开始渐涨了

进入到二月份&#xff0c;一切都将愈发生机盎然&#xff01;全国范围内的MBA/MEM/MPA项目都有各自的招生节奏和特点&#xff0c;提前批面试作为项目招考的重要方式之一&#xff0c;每年都会从年初开始陆续开放申请&#xff0c;而对于像浙大等名校来说&#xff0c;提前批面试的批…

mysql:数据库调优策略,sql调优

mysql&#xff1a;数据库调优策略。 硬件&#xff0c;系统配置&#xff0c;数据库表结构&#xff0c;sql及索引通过这些方面来优化项目的数据库层面。 越往后成本越低&#xff0c;但是效果确实越好。 第1步&#xff1a;选择适合的 DBMS第2步&#xff1a;优化表设计第3步&#…

【Java】面向对象笔记(下)

static关键字 static 静态 什么是静态 主要意义是在于创建独立于具体对象的域变量或者方法。以致于即使没有创建对象&#xff0c;也能使用属性和调用方法&#xff01; static关键字还有一个比较关键的作用就是用来形成静态代码块以优化程序性能。static块可以置于类中的任何…

数字授权如何满足工业软件多样化需求?

前言数字化转型的洪流正在不断对工业软件提出新的要求。在包括“智能工厂”、“智能生产”以及“智能物流”在内的主要领域里&#xff0c;工业软件正逐渐向智能化、嵌入式、分布式、互联化的方向演进。传统的软件保护和授权方式并不能适应工业软件新形式的需求。一方面&#xf…

蓝桥杯 stm32 RTC实时时钟

文章代码使用 HAL 库。 文章目录前言一、RTC 重要特性&#xff1a;二、CubeMX 创建工程。三、读取系统日期 函数。四、读取系统时间 函数。四、在 LCD 上显示 时间。总结实验效果前言 RTC (Real Time Clock)&#xff1a; 实时时钟。 RTC 模块拥有一个连续计数的 计数器&#…

mysql:有哪些索引,什么时候创建索引,什么时候不创建索引,创建索引的原则有哪些。

最近学习mysql&#xff0c;学习的索引的一些总结。 1.哪些索引 普通索引唯一性索引主键索引单列索引多列(组合、联合)索引全文索引补充&#xff1a;空间索引 小结&#xff1a;不同的存储引擎支持的索引类型也不一样 InnoDB &#xff1a;支持 B树。MyISAM &#xff1a; 支持…

基于JavaWeb的校园故障报修系统

项目描述 临近学期结束&#xff0c;还是毕业设计&#xff0c;你还在做java程序网络编程&#xff0c;期末作业&#xff0c;老师的作业要求觉得大了吗?不知道毕业设计该怎么办?网页功能的数量是否太多?没有合适的类型或系统?等等。这里根据疫情当下&#xff0c;你想解决的问…

Java里面为什么搞了双重检查锁,写完这篇文章终于真相大白了

双重检查锁定与延迟初始化 在 java 程序中&#xff0c;有时候可能需要推迟一些高开销的对象初始化操作&#xff0c;并且只有在使用这些对象时才进行初始化。此时程序员可能会采用延迟初始化。但要正确实现线程安全的延迟初始化需要一些技巧&#xff0c;否则很容易出现问题。比如…

城市POI数据爬取-百度地图版

1 API说明 目前百度地图的最新版为地图检索V2.0服务。详细介绍可以通过开发文档-web服务Api-地点检索V2.0获取。 在使用API前需要提前注册账号获取ak。对于免费账号&#xff1a;目前的每日访问次数是100次&#xff0c;最多可以获取2000条数据。 如不需讲解仅需要下载代码&am…

阿里软件架构师手写JDK源码,看完真的膜拜

最近有不少小伙伴在后台留言&#xff0c;说 Java 的面试越来越难了&#xff0c;尤其是技术面&#xff0c;考察得越来越细&#xff0c;越来越底层。 通过和大厂的面试官聊了一下发现&#xff0c;现在大厂特别爱考底层的一些原理&#xff0c;因为一些底层是不涉及到语言的&#x…

【Linux】进程状态的理解

&#x1f923; 爆笑教程 &#x1f449; 《看表情包学Linux》&#x1f448; 猛戳订阅 &#x1f525; &#x1f4ad; 写在前面&#xff1a;本章我们专门讲解进程的状态。我们先学习具体的 Linux 系统状态&#xff0c;再去介绍 OS 学科面对的概念如何理解 —— 运行态、终止态、阻…

二叉树的基础应用

二叉树 树概念及结构 1.1树的概念 树是一种非线性的数据结构&#xff0c;它是由n&#xff08;n>0&#xff09;个有限结点组成一个具有层次关系的集合。把它叫做树是因 为它看起来像一棵倒挂的树&#xff0c;也就是说它是根朝上&#xff0c;而叶朝下的。 有一个特殊的结点…

Seata分布式事务落地解决方案

引言上一篇文章介绍了分布式事务理论和相关解决方案的具体思路&#xff0c;我们下面快速复习一下相关知识点:1.分布式事务问题1.1.本地事务本地事务&#xff0c;也就是传统的单机事务。在传统数据库事务中&#xff0c;必须要满足四个原则&#xff1a;1.2.分布式事务分布式事务&…