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

【C++课程学习】:string的模拟实现

🎁个人主页:我们的五年

🔍系列专栏:C++课程学习

🎉欢迎大家点赞👍评论📝收藏⭐文章

 

目录

一.string的主体框架:

二.string的分析:

🍔构造函数和析构函数:

🍔迭代器:

🍔赋值运算符重载:

🍔空间扩容:

🍔尾插:

🍔获取_str,_capacity,_size:

 🍔[]重载,随机访问:

🍔在结尾追加字符串

🍔在指定位置插入和删除字符串:

🍔交换:

🍔获取子串和寻找元素和字符串:

🍔运算符重载:

🍔流插入和流提取重载:


通过对string的模拟实现,可以对string有更深的理解和认识,大概知道底层的逻辑,以后出现什么问题可以更好的解决。对学习以后的容器也有很大的帮助。

 

一.string的主体框架:

1.迭代器        2.运算符重载        3.

#define _CRT_SECURE_NO_WARNINGS 1
#pragma once
#include<iostream>
#include<cstring>
#include<assert.h>
#include<algorithm>

using namespace std;

namespace bit {
	class string {
	public:
		//迭代器
		typedef char* iterator;
		typedef const char* const_iterator;

		iterator begin();
		const_iterator begin()const;

		iterator end();
		const_iterator end()const;

		//构造函数
		string(size_t capacity = 4);
		string(const string& s);
		string(const char* str);
		string& operator=(const string& str);

		//析构函数
		~string();

		//异地扩容
		void reserve(size_t newcapacity);
		//尾插数据
		void push_back(char c);


		const char* c_str()const;

		size_t size()const;
		size_t capacity()const;

		char& operator[](size_t pos);
		const char& operator[](size_t pos)const;

		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 = 0, size_t len = npos);

		void swap(string& s);

		string substr(size_t pos = 0, size_t len = npos);

		size_t find(char ch, size_t pos = 0);
		size_t find(const char* str, size_t pos = 0);


		//运算符重载
		bool operator<(const string& s) const;
		bool operator>(const string& s) const;
		bool operator<=(const string& s) const;
		bool operator>=(const string& s) const;
		bool operator==(const string& s) const;
		bool operator!=(const string& s) const;

		//对sring进行清理
		void clear();
	private:
		const static size_t npos = -1;
		char* _str;
		//有效大小
		size_t _size;
		//空间的容量
		size_t _capacity;
	};

	//重载流插入和流提取
	istream& operator>> (istream& is, string& str);
	ostream& operator<< (ostream& os, const string& str);
}

二.string的分析:

🍔构造函数和析构函数:

构造函数常用的是三个,普通构造,拷贝构造,用const char*类型进行构造。

	//capacity表示要提前开的空间大小
	string::string(size_t capacity)
		:_str(new char[capacity + 1])
		, _capacity(capacity)
		, _size(0)
	{
		_str[0] = '\0';
	}

	//拷贝构造
	string::string(const string& s)
	{
		_capacity = s._capacity;
		_size = s._size;
		_str = new char[_capacity + 1];
		strcpy(_str, s._str);
	}

	string::string(const char* str)
	{
		size_t length = strlen(str);
		_str = new char[length + 1];
		_capacity = _size = length;

		strcpy(_str, str);
	}

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

🍔迭代器:

		//迭代器
		typedef char* iterator;
		typedef const char* const_iterator;

		iterator begin();
		const_iterator begin()const;

		iterator end();
		const_iterator end()const;

🍔赋值运算符重载:

	string& string::operator=(const string& str)
	{
		if (str._capacity > _capacity)
			reserve(str._capacity);
		strcpy(_str, str._str);
		return *this;
	}

🍔空间扩容:

	void string::reserve(size_t newcapacity)
	{
		//先判断是否需要扩容
		//如果newcapacity小于等于,就不会扩,也不会缩容
		if (newcapacity <= _capacity)
			return;
		char* p = new char[newcapacity + 1];
		_capacity = newcapacity;
		strcpy(p, _str);
		delete[] _str;
		_str = p;
	}

🍔尾插:

	void string::push_back(char c)
	{
		if (_size == _capacity)
		{
			//不能直接乘以两倍,_capacity可能为0,乘二还是为0
			size_t newcapacity = _capacity == 0 ? 4 : _capacity * 2;
			reserve(newcapacity);
		}
		_str[_size] = c;
		_size++;
	}

🍔获取_str,_capacity,_size:

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

	size_t string::size()const
	{
		return _size;
	}

	size_t string::capacity()const
	{
		return _capacity;
	}

 🍔[]重载,随机访问:

	char& string::operator[](size_t pos)
	{
		assert(pos >= 0 && pos < _size);
		return _str[pos];
	}

	const char& string::operator[](size_t pos)const
	{
		assert(pos >= 0 && pos < _size);
		return _str[pos];
	}

🍔在结尾追加字符串

	void string::append(const char* s)
	{
		size_t length = strlen(s);
		if(_size + length > _capacity)
			reserve(_size + length);
		//
		strcat(_str, s);
		_size += length;
	}

🍔在指定位置插入和删除字符串:

	void string::insert(size_t pos, char ch)
	{
		assert(pos >= 0 && pos <= _size);
		if (_size == _capacity)
			reserve(_capacity * 2);
		size_t end = _size + 1;
		while (end > pos)
		{
			_str[end] = _str[end - 1];
			end--;
		}
		_size++;
		_str[pos] = ch;
	}

	void string::insert(size_t pos, const char* str)
	{
		assert(pos >= 0 && pos <= _size);
		size_t length = strlen(str);
		if (length + _size > _capacity)
			reserve(length + _size);
		for (size_t i = _size - 1; i >= pos; i--)
			_str[i + length] = _str[i];
		_str[_size + length] = 0;
		for (size_t i = 0; i < length; i++)
			_str[i + pos] = str[i];
	}

	void string::erase(size_t pos, size_t len)
	{
		assert(pos >= 0 && pos < _size - 1);
		if (len>=_size-pos)
		{
			_str[pos] = '\0';
			_size = pos;
		}
		else
		{
			strcpy(_str + pos, _str + pos + len);
			_size -= len;
		}
	}

🍔交换:

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

🍔获取子串和寻找元素和字符串:

	string string::substr(size_t pos, size_t len)
	{
		assert(pos >= 0 && pos < _size);

		if (len > _size - pos)
		{
			bit::string substr(_str + pos);
			return substr;
		}
		else
		{
			bit::string substr(len);
			for (size_t i = pos; i < pos+len; i++)
				substr += _str[i];
			return substr;
		}
	}

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

	size_t string::find(const char* str, size_t pos)
	{
		char* p = strstr(_str + pos, str);
		return p - _str;
	}

🍔运算符重载:

	bool string::operator<(const string& s) const
	{
		return strcmp(_str, s._str)<0;
	}
	bool string::operator>(const string& s) const
	{
		return !((*this) <= s);
	}
	bool string::operator<=(const string& s) const
	{
		return ((*this) < s || (*this) == s);
	}
	bool string::operator>=(const string& s) const
	{
		return ((*this) > s || (*this) == s);
	}
	bool string::operator==(const string& s) const
	{
		return strcmp(_str, s._str) == 0;
	}
	bool string::operator!=(const string& s) const
	{
		return !((*this) == s);
	}

	string& string::operator+=(char ch)
	{
		insert(_size, ch);
		return *this;
	}

	string& string::operator+=(const char* str)
	{
		insert(_size, str);
		return *this;
	}

🍔流插入和流提取重载:

	istream& operator>>(istream& is, string& s)
	{
		s.clear();
		char it = is.get();
		while (it != ' ' && it != '\n')
		{
			s += it;
			it = is.get();
		}
		return is;
	}

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

 


http://www.kler.cn/a/391431.html

相关文章:

  • Python贪心
  • [AI部署-tensorRT] customlayer定义添加过程解析
  • 为深度学习创建PyTorch张量 - 最佳选项
  • WORD转PDF脚本文件
  • 如何通过openssl生成.crt和.key
  • 【BLE】CC2541之ADC
  • 两化融合评估流程
  • GPT模型发展放缓?《The Information》称是,OpenAI专家Noam Brown称否!
  • TDesign了解及使用
  • MySQL系列:一句SQL,MySQL是怎么工作的?
  • Linux SSH私钥认证结合cpolar内网穿透安全高效远程登录指南
  • 网站小程序app怎么查有没有备案?
  • uniapp上拉刷新下拉加载
  • Codeforces Round 984 (Div. 3) (A~E)
  • Python | Leetcode Python题解之第553题最优除法
  • Java poi 模板导出Word 带图片
  • React中右击出现自定弹窗
  • 了解外呼系统线路分类,提升业务效率
  • Django框架:Form组件及参数
  • ubuntu 22.04 镜像源更换
  • 51单片机使用NRF24L01进行2.4G无线通信
  • 系统架构设计师论文:大数据Lambda架构
  • 【JAVA基础】JVM双亲委派
  • Redis 内存突增时,如何定量分析其内存使用情况
  • Python OpenCV孤立点检测
  • Ansible内置模块之known_hosts