目录

一、什么是string类

二、string类的使用

1、string类对象的常见构造

2、string类对象的容量操作

3、string类对象的访问及遍历操作

4、string类对象的修改操作

5、string类非成员函数

6、其他函数


一、什么是string类

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.不能操作多字节或者变长字符的序列。

5. 在使用string类时,必须包含#include头文件以及using namespace std。

二、string类的使用

1、string类对象的常见构造

函数名称功能

string s;

构造空的string类对象,即空字符串

string s1(const char* s);

复制s指向的字符串

string s2(size_t n,char c);

生成n个c字符的字符串

string s3(const string&s);用string类对象拷贝构造

具体使用:

    string s;         // 构造空的string类对象s

	string s2("hello");//复制hello字符串给s2对象

	string s3(10, 'x');//生成10个x的字符串

	string s4(s2);     //用s2拷贝构造对象s4

2、string类对象的容量操作

函数功能
size返回字符串有效个数
length返回有效字符串个数
capacity返回总空间的大小
empty判断字符串是否为空
clear清空字符串
reserve为字符串预留空间(有点像malloc)
resize将有效的字符个数改成n个,多出的空间用字符c填充(有点像calloc)

具体使用: 

void test2()
{
	string s("hello");

	cout <<"size:"<< s.size() << endl;
	cout <<"length:"<< s.length() << endl;
	cout <<"capacity:"<< s.capacity() << endl;
	cout <<"empty:"<< s.empty() << endl;
}

结果如下: 

 对于clear函数,只是将s中的字符串清空,size置为0,但是不改变底层空间capacity的大小

void test2()
{
	string s("hello");

	cout <<"size:"<< s.size() << endl;
	cout <<"length:"<< s.length() << endl;
	cout <<"capacity:"<< s.capacity() << endl;
	cout <<"empty:"<< s.empty() << endl;
	cout << "----------" << endl;
	s.clear();
	cout << "size:" << s.size() << endl;
	cout << "capacity:" << s.capacity() << endl;
	cout << "empty:" << s.empty() << endl;

}

 我们再来详细看看resize

void resize (size_t n);
void resize (size_t n, char c);

①按照给定的字符扩容

void test3()
{
	string s("hello");
	s.resize(10, 'x');//将s中的有效字符个数增加到10个,后面多出的位置用x填充
	cout << "size:" << s.size() << endl;
	cout << "capacity:" << s.capacity() << endl;
	cout << s << endl;
}

结果如下: 

②默认给定的“\0”来填充扩容

void test4()
{
	string s("hello");
	s.resize(15);//将s中的有效字符个数增加到15个,后面多出的位置用"\0"填充
	cout << "size:" << s.size() << endl;
	cout << "capacity:" << s.capacity() << endl;
	cout << s << endl;
}

 结果如下:

③当是缩小大小时,会截断字符

void test5()
{
	string s("hello");
	s.resize(3);//将s中的有效字符个数减少到3个,只剩下原来的前三个
	cout << "size:" << s.size() << endl;
	cout << "capacity:" << s.capacity() << endl;
	cout << s << endl;
}

结果如下:

 我们再来详细看看reserve

void reserve (size_t n = 0);

①当n大于当前对象的容量时,将容量扩大到n或者大于n

void test6()
{
	string s;
	cout << "size:" << s.size() << endl;
	cout << "capacity:" << s.capacity() << endl;
	cout << "----------------" << endl;
	s.reserve(100);//将原来的capacity扩大到100
	cout << "size:" << s.size() << endl;
	cout << "capacity:" << s.capacity() << endl;
}

结果如下:

 ②当n小于容量时,不会缩小到容量

void test6()
{
	string s;
	cout << "size:" << s.size() << endl;
	cout << "capacity:" << s.capacity() << endl;
	cout << "----------------" << endl;
	s.reserve(100);//将原来的capacity扩大到100
	cout << "size:" << s.size() << endl;
	cout << "capacity:" << s.capacity() << endl;
	cout << "----------------" << endl;
	s.reserve(20);//不会将之前的capacity缩小
	cout << "size:" << s.size() << endl;
	cout << "capacity:" << s.capacity() << endl;
}

结果如下:

我们可以看到容量并没有缩小!!!

总结:

① resize(size_t n) 与 resize(size_t n, char c)都是将字符串中有效字符个数改变到n个,不同的是当字符个数增多时:resize(n)用0来填充多出的元素空间,resize(size_t n, char c)用字符c来填充多出的元素空间。注意:resize在改变元素个数时,如果是将元素个数增多,可能会改变底层容量的大小,如果是将元素个数减少,底层空间总大小不变。

② reserve(size_t res_arg=0):为string预留空间,不改变有效元素个数,当reserve的参数小于string的底层空间总大小时,reserve不会改变容量大小

3、string类对象的访问及遍历操作

函数功能
operator[ ](i)和s[i]

返回i位置的字符,const string类对象的调用

begin+endbegin获取第一个字符的迭代器,end获取最后一个字符的下一个位置的迭代器
rbegin+rendrbegin反向获取第一个字符的迭代器,rend反向获取最后一个字符的下一个位置的迭代器
范围forC++11支持更简洁的范围for遍历方式

operator[ ](i)

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

我们知道string类字符串的底层其实是数组实现的,于是我们可以采用下标的方式进行遍历。这里的operator[ ](i)和s[i]其实是完全一样的,只不过后者使用更加方便。他们返回对位于字符串中pos位置的字符的引用。

举例说明:

void test7()
{
	string s1("hello");
	const string s2("HELLO");
	cout << s1 << " " << s2 << endl;
	cout << s1[0] << " " << s2[0] << endl;

	s1[0] = 'x';
	cout << s1 << endl;

	//s2[0] = 'x';const类对象不能修改
}

三种遍历操作:(常用)

①operator[ ]的用法

void test8()
{	
	string s1("hello");
	string s2(s1);//拷贝构造
	for (size_t i = 0; i < s2.size(); ++i)
	{
		cout << s2.operator[](i) << " ";
		cout << s2[i] << " ";
	}
	cout << endl;
	for (size_t i = 0; i < s2.size(); ++i)
	{
		s2[i] += 1;//支持修改
	}
	cout << s2 << endl;
}

②迭代器

注意:这里暂时将迭代器想象成像指针一样的类型

void test9()
{
	string s("hello");
	//迭代器的用法
	string::iterator it = s.begin();
	while (it !=s.end())
	{
		cout << *it << endl;
		++it;
	}
}

③范围for

其实范围for的底层是将它给处理成了迭代器

void test10()
{
	string s("hello");
	for (auto ch : s)
		cout << ch << endl;
	//把s中的每个字符取出来,赋值给ch,不需要++,判断结束,自动往后迭代
}

4、string类对象的修改操作

函数功能
push_back在字符串后面尾插字符c
append在字符串后面追加字符串
operator+=在字符串后面追加字符串
c_str返回C格式字符串
find+npos从字符串pos位置开始,往后找字符c,返回该字符在字符串中的位置
rfind从字符串pos位置开始,往前找字符c,返回该字符在字符串中的位置
substr在字符串中,从pos位置开始,截取n个字符,然后将其返回

①push_back

void push_back (char c);
void test11()
{
	string s("xxx");
	cout << s << endl;
	s.push_back('H');
	s.push_back('E');
	s.push_back('L');
	s.push_back('L');
	s.push_back('O');
	cout << s << endl;//将HELLO依次拼接到xxx后面
}

结果如下:

②append 

string& append (const string& str);
string& append (const char* s);
void test12()
{
	string s1("hello");
	string s2("C++");
	s1.append(s2);//直接拼接一个对象字符串

	cout << s1 << endl;
	s1.append("haha");//直接拼接一个字符串
	cout << s1 << endl;
}

 ③operator+=(这是最常用的一个函数)

string& operator+= (const string& str);
string& operator+= (const char* s);
string& operator+= (char c);
void test13()
{
	string s1("hello");
	string s2("world");
	s1 += s2;
	cout << s1 << endl;

	s1 += "xxx";
	cout << s1 << endl;

	s1 += 'A';
	cout << s1 << endl;
}

结果如下: 

 ④c_str

const char* c_str() const;
void test14()
{
	string s("hello");
	//以语言的格式打印字符串
	cout << s.c_str() << endl;
}

结果如下:

 ⑤find函数

static const size_t npos = -1;//其实npos就是size_t的最大值

size_t find (const string& str, size_t pos = 0) const;
	
size_t find (const char* s, size_t pos = 0) const;
	
size_t find (const char* s, size_t pos, size_t n) const;
	
size_t find (char c, size_t pos = 0) const;
void test15()
{
	string s1("http://www.cplusplus.com/reference/string/string/find/");
	
	size_t pos1 = s1.find("com");//正向查找com字符串
	cout << pos1 << endl;

	string s2("http");
	size_t pos2 = s1.find(s2);//正向查找与s2对象匹配的字符串
	cout << pos2<< endl;

	size_t pos3 = s1.find('p');//正向查找字符p
	cout << pos3 << endl;
}

 结果展示:

 ⑥rfind

size_t rfind (const string& str, size_t pos = npos) const;
	
size_t rfind (const char* s, size_t pos = npos) const;

size_t rfind (const char* s, size_t pos, size_t n) const;
	
size_t rfind (char c, size_t pos = npos) const;
void test16()
{
	string s1("http://www.cplusplus.com/reference/string/string/find/");

	size_t pos1 = s1.rfind("string");//反向查找是s1中第一次出现的string的首位置
	cout << pos1 << endl;

	string s2("cplusplus");
	size_t pos2 = s1.rfind(s2);//反向查找s1中与s2对象匹配的首位置
	cout << pos2 << endl;

	size_t pos3 = s1.rfind('h'); //反向查找s1中h字符
	cout << pos3 << endl;
}

 结果展示:

⑦substr

string substr (size_t pos = 0, size_t len = npos) const;
void test17()
{
	//获取file的后缀
	string file("string.cpp");
	size_t pos = file.rfind('.');//反向查找 . 点
	string suffix(file.substr(pos,file.size()-pos));//从点开始,包括点的后面部分
	cout << suffix << endl;
}

结果:

5、string类非成员函数

函数

功能

operator+尽量少用,因为是传值返回,导致拷贝效率低
operator>>流插入运算符重载
operator<<流提取运算符重载
getline获取一行字符串
relational operators大小比较

输入输出问题: 

void test18()
{
	string s;
	cin >> s;//输入hello world
	cout << s << endl;//只会打印hello
}

 我们知道cin标准输入,当从键盘上面读取到空格或者\n时,会停止读取,这时候就要用到getline函数了

istream& getline (istream& is, string& str, char delim);
//从is中读取字符串,储存在str对象中,直到遇到delim界限符或者\n停止

istream& getline (istream& is, string& str);
//从is中读取字符串,储存在str对象中,直到遇到\n停止
void test18()
{
	string s1;
	getline(cin,s1,'e');//输入hello world
	cout << s1 << endl;//输出h
}
void test18
{
	string s2;
	getline(cin, s2);//输入hello world
	cout << s2 << endl;//输出hello world
}

大小比较问题:

void test19()
{
	string s1("hello");
	string s2("string");
	cout << (s1 > s2) << endl;//输出0
}

这里的有点不一样,比较的是第一个字符的ASCII码值的大小,若第一个相等则比较第二个,返回一个bool值,以此类推。

6、其他函数

做OJ题的时候比较常用

stoi——字符串转为整型

void test20()
{
	int val = stoi("12345");
	cout << val << endl;
}

还有很多类似的函数: stod stoll stof等等,感兴趣可以自行了解

to_string——其他类型转为字符串

void test21()
{
	string str = to_string(3.14);
	cout << str << endl;
}

谢谢观看!

Logo

权威|前沿|技术|干货|国内首个API全生命周期开发者社区

更多推荐