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

[C++] string类的介绍与构造的模拟实现,进来看吧,里面有空调

在这里插入图片描述

文章目录

  • 1、string类的出现
    • 1.1 C语言中的字符串
  • 2、标准库中的string类
    • 2.1 string类
  • 3、string类的常见接口说明及模拟实现
    • 3.1 string的常见构造
    • 3.2 string的构造函数
    • 3.3 string的拷贝构造
    • 3.4 string的赋值构造
  • 4、完整代码

1、string类的出现

1.1 C语言中的字符串

C语言中,字符串是以’\0’结尾的一些字符的集合,为了操作方便,C标准库中提供了一些str系列的库函数,但是这些库函数与字符串是分离开的,不太符合面向对象的思想,而且底层空间需要用户自己管理,稍不留神可能还会越界访问。
因此C++中,为了让我们更简单、方便、快捷的使用字符串类型,C++提供了string类。

2、标准库中的string类

2.1 string类

string类文档介绍
string类在使用的时候我们需要包含头文件 #include ,以及using namespace std;

  1. 字符串是表示字符序列的类。
  2. 标准的字符串类提供了对此类对象的支持,其接口类似于标准字符容器的接口,但添加了专门用于操作单字节字符字符串的设计特性。
  3. string类是使用char(即作为它的字符类型,使用它的默认char_traits和分配器类型(关于模板的更多信息,请参阅basic_string)。
  4. string类是basic_string模板类的一个实例,它使用char来实例化basic_string模板类,并用char_traits和allocator作为basic_string的默认参数(根于更多的模板信息请参考basic_string)。
  5. 注意,这个类独立于所使用的编码来处理字节:如果用来处理多字节或变长字符(如UTF-8)的序列,这个类的所有成员(如长度或大小)以及它的迭代器,将仍然按照字节(而不是实际编码的字符)来操作。

总结:
1. string是表示字符串的字符串类。
2. 该类的接口与常规容器的接口基本相同,再添加了一些专门用来操作string的常规操作。
3. string在底层实际是:basic_string模板类的别名,
typedef basic_string<char, char_traits, allocator>string;
4. 不能操作多字节或者变长字符的序列。

3、string类的常见接口说明及模拟实现

3.1 string的常见构造

(constructor)函数名称功能说明
string()构造空的string类对象,即空字符串
string(const char* s)用C-string来构造string类对象
string(size_t n, char c)string类对象中包含n个字符c
string(const string&s)拷贝构造函数
int main()
{string s1; // 构造空的string类对象s1string s2("hello"); // 用C格式字符串构造string类对象s2string s3(s2); // 拷贝构造s3return 0;
}

3.2 string的构造函数

namespace s
{class string{public://string()//	:_str(new char[1]{ '\0' })//这里开一个空间,并赋值为\0,如果直接赋值为nullptr的话//	,_size(0)				  //在实例化一个无参对象后,对对象进行操作的时候会引发空指针异常//	,_capacity(0)//{}//全缺省构造函数string(const char* str = "")//常量字符串默认以\0结尾,因此不需要给: _size(strlen(str)), _capacity(_size){_str = new char[_capacity + 1];//多开一个空间,需要存放\0strcpy(_str, str);//将字符串拷贝到_str}private:char* _str;size_t _size;//有效存储字符个数size_t _capacity;//实际空间};
}

这里我们用一个全缺省将无参构造与构造都包括了。
这个构造函数很简单,这里就不多讲了,大家应该可以看懂。

3.3 string的拷贝构造

传统写法:

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

现代写法:

void swap(string& s)
{std::swap(_str, s._str);std::swap(_size, s._size);std::swap(_capacity, s._capacity);
}
//现代写法 s1(s2)
string(const string& s):_str(nullptr),_size(0),_capacity(0)
{string tmp(s._str);//构造swap(tmp);//this->swap(s);
}

库里提供了交换函数,我们封装一个交换函数,内部的实现就使用库里面的swap,我们拷贝构造函数里面先调用构造函数,实例化一个tmp,再将tmp与this交换。

3.4 string的赋值构造

传统写法:

string& operator=(const string& s)
{if (this != &s){string tmp(s);//拷贝构造swap(tmp);}return *this;
}

现代写法:这里与拷贝构造的现代写法是类似的,不再多说。

void swap(string& s)
{std::swap(_str, s._str);std::swap(_size, s._size);std::swap(_capacity, s._capacity);
}
string& operator=(const string& s)
{if (this != &s){string tmp(s);//拷贝构造swap(tmp);}return *this;
}

我们其实可以对现代写法继续精简一下,我们这里参数是引用传参,我们不用引用就会产生一次拷贝,对拷贝的形参直接进行交换,这样就可以简化写法,但是这里的效率其实是一样的,都是一次拷贝构造 + 一次交换。

// 极致写法,效率与上面一样
string& operator=(string tmp)//参数直接调用拷贝构造
{swap(tmp);return *this;
}

4、完整代码

#include <string.h>
#include <algorithm>
using namespace std;namespace s
{class string{public://string()//	:_str(new char[1]{ '\0' })//这里开一个空间,并赋值为\0,如果直接赋值为nullptr的话//	,_size(0)				  //在实例化一个无参对象后,对对象进行操作的时候会引发空指针异常//	,_capacity(0)//{}//全缺省构造函数string(const char* str = "")//常量字符串默认以\0结尾,因此不需要给:_size(strlen(str)),_capacity(_size){_str = new char[_capacity + 1];//多开一个空间,需要存放\0strcpy(_str, str);//将字符串拷贝到_str}//拷贝构造// 传统写法//string(const string& s)//	:_size(s._size)//	,_capacity(s._capacity)//{//	_str = new char[s._capacity + 1];//	strcpy(_str, s._str);//}void swap(string& s){std::swap(_str, s._str);std::swap(_size, s._size);std::swap(_capacity, s._capacity);}//现代写法 s1(s2)string(const string& s):_str(nullptr),_size(0),_capacity(0){string tmp(s._str);//构造swap(tmp);//this->swap(s);}//赋值// 传统写法//string& operator=(const string& s)//{//	if (this != &s)//	{//		char* tmp = new char[s._capacity];//		strcpy(tmp, s._str);//		delete[] _str;//		_str = tmp;//将tmp字符串直接给_str,类似浅拷贝//		_size = s._size;//		_capacity = s._capacity;//	}//	return *this;//}//现代写法//string& operator=(const string& s)//{//	if (this != &s)//	{//		string tmp(s);//拷贝构造//		swap(tmp);//	}//	return *this;//}// 极致写法,效率与上面一样string& operator=(string tmp)//参数直接调用拷贝构造{swap(tmp);return *this;}~string(){cout << "~string()" << endl;delete[] _str;_size = _capacity = 0;}private:char* _str;size_t _size;//有效存储字符个数size_t _capacity;//实际空间};
}
http://www.lryc.cn/news/127362.html

相关文章:

  • 【Apollo】赋能移动性:阿波罗自动驾驶系统的影响
  • Camunda 7.x 系列【19】表达式语言
  • 详解RFC 793文档-4
  • ubuntu16编译内核源码并替换
  • AI 绘画Stable Diffusion 研究(九)sd图生图功能详解-老照片高清修复放大
  • Linux系统安装Google Chrome
  • 带你了解SpringBoot支持的复杂参数--自定义对象参数-自动封装
  • Raspberry Pi Pico RP2040制作低成本FPGA JTAG工具
  • 【工具】Python从临时邮箱获取验证码
  • 时序预测 | MATLAB实现基于GRU门控循环单元的时间序列预测-递归预测未来(多指标评价)
  • 在本地搭建WAMP服务器并通过端口实现局域网访问(无需公网IP)
  • Redis之删除策略
  • SpringBoot的配置文件以及日志设置
  • 【Java集合框架面试题(30道)】
  • Android 组件
  • kotlin的数据类型和类型转换
  • 常见架构类型
  • 搭建Web服务器并用cpolar发布至公网访问
  • NanoPi NEO移植LVGL8.3.5到1.69寸ST7789V屏幕
  • 阿里云服务区ECS,申请免费的服务器之后,如何使用xshell 登陆,找不到匹配的host key 算法
  • 【CTF-web】备份是个好习惯(查找备份文件、双写绕过、md5加密绕过)
  • C#和Java的大端位和小端位的问题
  • 如何在 iOS 上安装并使用 ONLYOFFICE 文档
  • 【AI】如何用AI生成XMind思维导图
  • k8s问题汇总
  • 打造专属照片分享平台:快速上手Piwigo网页搭建
  • OpenAI全球招外包大军,手把手训练ChatGPT取代码农 ; 码农:我自己「杀」自己
  • 船舶法兰盘法兰管件3D扫描尺寸测量|三维扫描检测|CAV测量-CASAIM
  • 每天一道leetcode:1218. 最长定差子序列(动态规划中等)
  • C#的 Settings.Settings配置文件的使用方法