【C++】string类(上)

avatar
作者
猴君
阅读量:0

在这里插入图片描述

个人主页~


string

一、标准库中的string类

1、什么是string类

(1)字符串是表示字符序列的类,string是表示字符串的字符串类
(2)标准的字符串提供了对此类对象的支持,其接口类似于标准字符容器的接口与常规容器的接口基本相同,但添加了专门用于操作单字节字符字符串的设计特性,也就是专门用来操作string的常规操作
(3)string类是使用char作为它的字符类型
(4)string类是basic_string模版类的一个实例,它使用char来实例化basic_string模板类,并用char_traits和allocator作为basic_string的默认参数
在这里插入图片描述

basic_string<char> s1; string s2; //这两个是一样的,string就是basic_string的char类型特化 typedef basic_string<char, char_traits, allocator> string;//底层 

(5)不能操作多字节或者变长字符的序列
(6)在使用时要包头文件以及展开命名空间

2、string类的常用接口讲解

(1)string类的常见构造

函数名称功能说明
string()构造空的字符串
string(const char* s)用C格式的字符串构造字符串
string(const string& s)拷贝构造函数
string(size_t n,char c)字符串中包含n个字符c
void test() { 	string s1; 	string s2("hello world"); 	string s3(s2); 	string s4(5, 'a'); } 

在这里插入图片描述

(2)string类的容量操作

函数名称功能说明
size返回字符串有效字符长度
empty检测字符串是否为空,是返回true,否返回false
clear清空有效字符
reserve为字符串预留空间
resize将有效字符个数改为n个,多出的空间用字符c填充
capacity总空间大小
length返回字符串有效字符长度
void test2() { 	string s1("hello world");  	cout << s1 << endl; 	//测试size和length 	cout << s1.size() << endl; 	cout << s1.length() << endl; 	//size和length底层实现原理完全相同,在刚刚创造出string的时候,对于字符串来说,叫length很合适, 	//所以起名为length,但不久后STL产生了,为了与其他的模版比如list,vector等统一, 	//所以加了size,保持接口一致性,一般都用size 	 	//测empty 	cout << s1.empty() << endl; 	 	//测capacity 	cout << s1.capacity() << endl; 	 	//测resize 	s1.resize(5, 'a'); 	cout << s1 << endl; 	s1.resize(15, 'a'); 	cout << s1 << endl; 	//resize(size_t n)与resize(size_t n,char c)都是将字符串中有效字符个数改到n个, 	//不同的是当字符个数增多时,resize(size_t n)用0来填充多出的元素空间, 	//resize(size_t n,char c)用c字符填充多出的元素空间 	//resize改变元素个数时,如果个数增多,可能会改变底层容量的大小,如果减少则不变 	 	//测reserve 	s1.reserve(15); 	cout << s1.capacity() << endl; 	//为string预留空间,如果reserve的参数小于string的底层空间总大小时,reserve不会改变容量大小 	 	//测clear 	s1.clear(); 	cout << s1.capacity() << endl; 	s1 += "little monster";//+=在后边有讲解 	cout << s1 << endl; 	//clear只是将string中有效字符清空,不改变底层大小,可以再键入新内容 } 

在这里插入图片描述

(3)string类对象的访问及遍历

函数名称功能说明
operator[ ]返回pos位置的字符,const string类调用
begin 和 endbegin获取一个字符的迭代器,end获取最后一个字符的后一个位置的迭代器
rbegin 和 rendrbegin获取一个字符的迭代器,rend获取最后一个字符的后一个位置的迭代器
范围for更简洁的遍历
void test3() { 	string s1("hello world");  	cout << s1[6] << endl; 	//使用下标操作符[]可以直接找到对应位置的字符 	 	//string::iterator it = s1.begin(); 	auto it = s1.begin();//这里体现出了auto的优越性,上面一长串的类型可以直接用auto推导代替 	for (it; it != s1.end(); it++) 	{ 		cout << *it << " "; 	} 	cout << endl; //begin记录第一个有效字符的位置,end记录最后一个有效字符的后一个位置 //从头到尾的打印,++是一个重载运算符,在其他类中,包括链表类也可以直接找到下一个成员  	for (auto it : s1) 	{ 		cout << it << " "; 	} 	cout << endl; //使用范围for进行遍历,十分方便  	auto rit = s1.rbegin(); 	for (rit; rit != s1.rend(); rit++) 	{ 		cout << *rit << " "; 	} //rbegin和rend是一对与begin和end相反的函数,rbegin记录的是最后一个有效字符的位置,rend记录的是第一个有效字符的前一个位置,通过++实现逆向输出 } 

在这里插入图片描述

(4)string类对象的修改

函数名称功能说明
operator+=在字符串后追加字符串
c_str返回C格式字符串
find从字符串pos位置开始往后找字符c,返回该字符在字符串中的位置
push_back在字符串后尾插字符
append在字符串后追加一个字符串
rfind从字符串pos位置开始往前找字符c,返回该字符在字符串中的位置
substr在str中从pos位置开始,截取n个字符,然后将其返回
void test4() { 	string s1("hello world"); 	 	//+=测试 	s1 += "!!!!"; 	cout << s1 << endl; 	 	//c_str测试 	cout <<s1.c_str() << endl; 	//c_str就是将c++的格式转化为c语言的格式,这样字符串就可以用C语言的方式来操作 	//并且c_str返回的是指针,因为<<是重载运算符,所以才显示指向的内容 	 	//find测试 	cout << s1.find('w',2) << endl; 	 	//pushback测试 	s1.push_back('6'); 	cout << s1 << endl; 	 	//append测试 	s1.append("789"); 	cout << s1 << endl; 	 	//rfind测试 	cout << s1.rfind('l' , 6) << endl; 	 	//substr测试 	cout << s1.substr(0, 11) << endl; 	//substr只会截取并返回,不改变s1的内容 } 

在这里插入图片描述
在string尾部追加字符时,可以使用push_back , append , += ,push_back只能追加字符,append追加字符串,所以我们一般常用的是+=,既可以追加字符又可以追加字符串,并且使用起来书写简单,代码可读性高

对string进行操作时,如果可预见可以放多少字符,可以用reserve把空间预留好

这里的 find 和 rfind 与 begin rbegin 那一套一样,r都表示相反,注意这里find是从pos开始往后找到的第一个指定字符,rfind是从pos开始往前找到的第一个指定字符,然后返回该字符所在位置的下标

(5)string类非成员函数

函数名称功能说明
operator>>输入运算符重载
operator<<输出运算符重载
getline获取一行字符串
relational operators大小比较
operator+少用,传值返回,深拷贝效率低

这部分内容比较简单,我设置了超链接,直接点进去看一下文档就可以了

(6)其他

string类还有很多其他的操作,不一一列举了,需要时直接打开cplusplus查找文档即可
string类
在这里插入图片描述
在这里插入图片描述

(7)vs和g++下string结构说明

前提:32位平台

vs下的string结构

string总共占28个字节,内部结构稍微复杂一点,有一个联合体,用来定义string中字符串的存储空间:当字符串长度小于16时,使用内部固定的字符数组存放,当字符串长度大于等于16时,从堆上开辟空间,这样保证了字符串在较小时不需要通过堆创建,提高了效率,占16字节

还有一个size_t字段保存字符串长度,占4字节,一个size_t字段保存从堆上开辟空间总的容量,占4字节

最后还有一个指针,4字节,共28字节

g++下string结构

在g++下,string通过写时拷贝实现,只占4个字节,内部只包含一个指针,指向一块堆空间,堆空间内部包含了空间总大小、字符串有效长度、引用计数

写时拷贝:在数据第一次写入到某个存储位置时,首先将原有内容拷贝出来,写到另一位置处,然后再将数据写入到存储设备中,该技术只拷贝在拷贝初始化开始之后修改过的数据
简单来说就是在用之前不开空间,在真正要修改和写入时才开辟空间,可以减少空间的浪费,它是在浅拷贝的基础上增加了引用计数的方式实现的

引用计数:用来记录资源使用者的个数,在构造时,将资源的计数给成1,每增加一个对象使用该资源,就给计数增加1,当某个对象被销毁时,先给该计数减1,然后再检查是否需要释放资源,如果计数为1,说明该对象是该资源的最后一个使用者,将该资源释放,否则因为其他对象还在使用该资源,该资源就不能释放


今日分享就到这里~

在这里插入图片描述

    广告一刻

    为您即时展示最新活动产品广告消息,让您随时掌握产品活动新动态!