【C++初阶】string的模拟实现
创始人
2024-04-29 01:30:19
0

文章目录

  • string的介绍
  • string的模拟实现
    • string类的成员变量
    • Member functions
      • constructor(构造函数)
      • destructor(析构函数)
      • operator=(给创建出来的对象赋值)
    • Iterators(迭代器)
      • begin
      • end
    • Capacity
      • size
      • capacity
      • reserve
      • resize
      • clear
    • Element access
      • operator[]
    • Modifiers
      • operator+=
      • append
      • push_back
      • insert
      • erase
      • swap
    • String operations
      • c_str
      • find
    • Member constants
      • npos
    • Non-member function overloads
      • operator>>
      • operator<<
    • 比较运算符重载
  • 完整版string类代码
    • string.h
    • test.cpp

string的介绍

string是C++ STL库中一个重要的容器,它分为以下几个部分(将在vs编译器下实现)
Member function
在这里插入图片描述
Iterators
在这里插入图片描述
Capacity
在这里插入图片描述
Element access
在这里插入图片描述
Modifiers
在这里插入图片描述
String operations
在这里插入图片描述
Member constants
在这里插入图片描述
Non-member function overloads
在这里插入图片描述
在大家了解了string的每个部分的成员函数之后,我们就要去模拟实现每个部分之中在日常中经常使用到的一些成员函数。

string的模拟实现

string类的成员变量

在这里插入图片描述

Member functions

constructor(构造函数)

在这里插入图片描述
在这里我们主要实现构造函数常用的三种也就是图中的(1),(2),(4)
在这里插入图片描述
在这里插入图片描述
其中一写函数的注释代码,是代码优化前的写法,大家可以用来做参考和理解。
在这里插入图片描述

destructor(析构函数)

在这里插入图片描述
在这里插入图片描述

operator=(给创建出来的对象赋值)

在这里插入图片描述
在这里我们只实现第一种。
在这里插入图片描述
在这里插入图片描述

Iterators(迭代器)

在这里我们只实现了正向迭代器
在这里插入图片描述
由图片可以看出起始迭代器就只是一个被重新定义的指针。

begin

在这里插入图片描述
在这里插入图片描述

end

在这里插入图片描述

在这里插入图片描述
end和begin相似

Capacity

size

在这里插入图片描述
在这里插入图片描述

capacity

在这里插入图片描述
在这里插入图片描述

reserve

在这里插入图片描述
在这里插入图片描述

resize

在这里插入图片描述
在这里插入图片描述
resize功能和reserve一样但是多了个初始化和缩容
在这里插入图片描述

clear

在这里插入图片描述
在这里插入图片描述

Element access

operator[]

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
都是返回pos位置的字符

Modifiers

operator+=

在这里插入图片描述
在这里插入图片描述
这里只实现了(2)和(3)两个版本

append

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

push_back

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

insert

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里指定位置插入函数就实现了2个常用的版本

erase

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

swap

在这里插入图片描述
在这里插入图片描述

String operations

c_str

在这里插入图片描述
在这里插入图片描述

find

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

Member constants

npos

在这里插入图片描述

Non-member function overloads

operator>>

在这里插入图片描述
在这里插入图片描述

operator<<

在这里插入图片描述
在这里插入图片描述

比较运算符重载

在这里插入图片描述

完整版string类代码

string.h

#pragma once
namespace lzy
{class string{public:typedef char* iterator;typedef const char* const_iterator;iterator begin(){return _str;}const_iterator begin()const{return _str;}iterator end(){return _str + _size;}const_iterator end()const{return _str + _size;}//string sstring():_str(new char[1]), _size(0), _capacity(0){_str[0] = '\0';}//创建一个空类//string("hello world")string(const char* s):_size(strlen(s)), _capacity(_size){_str = new char[_size + 1];strcpy(_str, s);}//用字符串初始化创建一个类//string s1(s2)string(const string& s)//:_size(s._size):_size(0), _capacity(0){/*_str = new char[_size+1];strcpy(_str, s._str);*/string temp(s._str);swap(temp);}//用类初始化创建一个类~string(){delete[] _str;}//s1 = s2 = s3//s1 = s1void swap(string& s){std::swap(_str, s._str);std::swap(_size, s._size);std::swap(_capacity, s._capacity);}//const string& operator=(const string& s)//{//	if (&s != this)//	{//		//重新为数组开辟一块空间,不管空间是否足够//		delete[] _str;//		_str = new char[s._size + 1];//		//开始为这块新开辟的空间赋值//		for (int i = 0; i <= s._size; i++)//		{//			_str[i] = s._str[i];//		}//		_size = s._size;//		_capacity = s._capacity;//	}//	return *this;//}const string& operator=(string s){//if (&s != this)//{//	//重新为数组开辟一块空间,不管空间是否足够//	delete[] _str;//	_str = new char[s._size + 1];//	//开始为这块新开辟的空间赋值//	for (int i = 0; i <= s._size; i++)//	{//		_str[i] = s._str[i];//	}//	_size = s._size;//	_capacity = s._capacity;//}/*swap(_str, s._str);swap(_size, s._size);swap(_str, s._str);*/swap(s);return *this;}const char* c_str() const{return _str;}size_t size()const{return _size;}size_t capacity()const{return _capacity;}//s[i] 可修改char& operator[](size_t pos){//越界访问就报错assert(pos < _size);return *(_str + pos);}//s[i] const对象不可修改const char& operator[](size_t pos)const{// 越界访问就报错assert(pos < _size);return *(_str + pos);}void reserve(size_t n){if (n > _capacity){char* temp = new char[n + 1];strcpy(temp, _str);delete[] _str;_str = temp;_capacity = n;}}void resize(size_t n, char x = '\0'){if (n > _capacity){char* temp = new char[n + 1];memset(temp, x, n + 1);strcpy(temp, _str);delete[] _str;_str = temp;_capacity = n;_size = n;}else{_size = n;_str[_size] = '\0';}}//尾插入一个字符void push_back(char ch){if (_size == _capacity){//增容/*size_t newcapacity = _capacity == 0 ? 4 : _capacity * 2;char* temp= new char[newcapacity+1];strcpy(temp, _str);delete[] _str;_str = temp;_capacity = newcapacity;*/reserve(_capacity == 0 ? 4 : _capacity * 2);}_str[_size] = ch;_size++;_str[_size] = '\0';}//尾插入一段字符串void append(const char* s){size_t length = strlen(s);//判断空间是否足够if (_size + length > _capacity){增容//char* temp = new char[_size + length+1];//strcpy(temp, _str);//delete[] _str;//_str = temp;//_capacity = _size;///*for (int i = 0; i //	_str[i + _size] = s[i];//}*///	//增容reserve(_size + length);//	for (int i = 0; i < length; i++)//	{//		_str[i + _size] = s[i];//	}//	_size += length;//	//}//else//{//	for (int i = 0; i //		_str[i + _size] = s[i];//	}//	_size += length;//	_str[_size] = '\0';}strcpy(_str + _size, s);_size += length;}string& operator+=(char ch){push_back(ch);return *this;}string& operator+=(const char* s){append(s);return *this;}size_t find(char ch) const{for (size_t i = 0; i < _size; i++){if (_str[i] == ch){return i;}}return npos;}size_t find(const char* s, size_t pos = 0){char* temp = strstr(_str + pos, s);if (temp == nullptr){return npos;}return temp - _str;}string& insert(char ch, size_t pos){if (_size == _capacity){//增容/*size_t newcapacity = _capacity == 0 ? 4 : _capacity * 2;char* temp = new char[newcapacity + 1];strcpy(temp, _str);delete[] _str;_str = temp;_capacity = newcapacity;*/reserve(_capacity == 0 ? 4 : _capacity * 2);}size_t end = _size + 1;while (pos < end){_str[end] = _str[end - 1];--end;}_str[pos] = ch;_size++;//_str[_size] = '\0';return *this;}string& insert(const char* s, size_t pos){assert(pos <= _size);size_t len = strlen(s);if (_size + len > _capacity){//增容/*size_t newcapacity = _size + len;char* temp = new char[newcapacity + 1];strcpy(temp, _str);delete[] _str;_str = temp;_capacity = newcapacity;*/reserve(_size + len);}size_t end = _size + len;while (pos + len < end){_str[end] = _str[end - len];--end;}//_str[pos] = ch;strcpy(_str + pos, s);_size += len;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;}else{//_str[pos] = '\0';strcpy(_str + pos, _str + pos + len);_size -= len;}return *this;}void clear(){_str[0] = '\0';_size = 0;}private:char* _str;size_t _size;size_t _capacity;static size_t npos;};size_t string::npos = -1;bool operator<(const string& s1, const string& s2){size_t begin1 = 0, begin2 = 0;while (begin1 < s1.size() && begin2 < s2.size()){if (s1[begin1] > s2[begin2]){return false;}else if (s1[begin1] < s2[begin2]){return true;}else{begin1++;begin2++;}}return begin2 < s2.size() ? true : false;}bool operator==(const string& s1, const string& s2){return strcmp(s1.c_str(), s2.c_str());}bool operator>(const string& s1, const string& s2){return !(s1 < s2 || s1 == s2);}bool operator<=(const string& s1, const string& s2){return !(s1 > s2);}bool operator>=(const string& s1, const string& s2){return !(s1 < s2);}ostream& operator<<(ostream& out, const string& s){///*out << s.c_str();*/不能这么写for (auto e : s){out << e;}return out;}istream& operator>>(istream& in, string& s){s.clear();char ch = in.get();while (ch != '\n'){//s += ch;s += ch;ch = in.get();}return in;}void test_string1(){string s1;string s2("hello world");string s3(s2);s1 = s2;s2 = s2;cout << s1.c_str() << endl;cout << s1.size() << endl;//cout << s1[s1.size() - 1] << endl;cout << s1[s1.size()] << endl;}void test_string2(){string s1;s1.push_back('H');s1.append("ell");s1.append("o world");string::iterator it = s1.begin();while (it != s1.end()){cout << *it << " ";it++;}//s1.reserve(1000);//s1.resize(100);/*s1 += ' ';s1 += "hello mom";*/}void test_string3(){string s1;/*s1.push_back('H');s1.append("ell");s1.append("o world");*/s1.insert('H', s1.size());s1.insert("ell", s1.size());s1.insert("o world", s1.size());string::iterator it = s1.begin();while (it != s1.end()){cout << *it << " ";it++;}cout << endl;cout << s1.find('H') << endl;cout << s1.find("world") << endl;//s1.erase(0);s1.erase(5, 1);//s1.erase(0, 6);it = s1.begin();while (it != s1.end()){cout << *it << " ";it++;}cout << endl;/*string s2(s1);it = s2.begin();while (it != s2.end()){cout << *it << " ";it++;}cout << endl;*/}void test_string4(){string s1("abcd");//string s2("abcd");string s3("abcde");/*cout << (s1 < s2) << endl;cout << (s1 < s3) << endl;*//*cout << (s1 == s3) << endl;cout << (s1 == s2) << endl;*//*s1 += '\0';s1 += "hello";*//*cout << s1 << endl;string s2;cin >> s2;cout << s2;*///s1.clear();cout << s1;}}

test.cpp

#define _CRT_SECURE_NO_WARNINGS 1
#include
using namespace std;
#include
#include
#include"string.h"
int main()
{lzy::test_string1();//lzy::test_string2();//lzy::test_string3();//lzy::test_string4();return 0;
}

vs编译器下的实现

相关内容

热门资讯

学习手记丨制度治权、依规用权 在二十届中央纪委五次全会上,习近平总书记指出:“党的自我革命重在治权,把权力关进制度笼子是新时代全面...
疑局长和女下属存“不正当关系”... 认为局长丁某在工作上对其“不公”,科长苏某向周某1表达了不满。因怀疑丁某和女下属肖某存在“不正当关系...
34岁男子称与57岁女子交往一... 近日,长沙的刘先生向九派新闻反映,2021年,其通过一起打球的朋友介绍结识了周女士,两人相处两三个月...
罗永浩需要为西贝预制菜风波担责... 16日下午,贾国龙更新微博称:“今晚(16日)10点将就罗永浩对西贝的重大污蔑诽谤一一全面回应”,并...
广州打造“穗育苗”品牌“守护少... 近年来,广州市认真落实党中央“强化未成年人违法犯罪预防和治理”决策部署及省委、省政府工作要求,进一步...
今日谈:政策托底,为生育护航 本文转自【人民日报】; 原标题:政策托底,为生育护航(今日谈) 继育儿补贴制度全面落地实施后,近日,...
学生上午匿名投诉学校下午就被找... 1月17日,云浮市伊顿实验学校发布情况通报: 近期,网传有关我校“一学生上午匿名投诉学校下午就被找到...
中国银行招标结果:2025-2... 证券之星消息,根据天眼查APP-财产线索数据整理,中国银行股份有限公司1月14日发布《2025-20...
海南建立失能老年人养老服务补贴... 人民网海口1月17日电(刘杨)为实施积极应对人口老龄化国家战略,加强失能老年人照护服务,提振养老服务...
女子连续五天向出轨丈夫公开“道... 封面新闻记者 宋潇 自1月12日起,河南省三门峡市女子牛某某连续五天在社交媒体平台发布向丈夫“道歉”...