string类虽然不在STL的容器中,但string类十分重要,string类是对字符串的存储和相关操作。
std::basic_string类是C++的一个模版类,它支持多种字符类型。
- char :用于表示一个字节的字符,使用ASCII编码。
- wchar_t :用于表示宽字符,可以支持更广泛的字符集。
- char8_t :用于表示8位的Unicode字符(UTF-8)。
- char16_t :用于表示16位的Unicode字符(UTF-16)。
- char32_t :用于表示32位的Unicode字符(UTF-32)。
basic_string 实例化出来的4种类型:
- string
- wstring
- u16string
- u32string
string类是basic_string模版类的一个实例,它使用char来实例化basic_string模版类,并使用char_traits和allocator 作为basic_string的默认参数。
1、string类是表示字符序列的类。
2、标准的字符串类提供了对此对象的支持,其接口类似于标准字符容器的接口,但是添加了专门用于操作单字节字符字符串的设计特性。
3、string类封装了字符数组,并提供了一系列成员函数来执行字符串的创建、修改、连接、查找、替换等操作。
string类出现实际上是比STL要早的,后来才分入STL库里面;所以,string一开始设计比较冗余,有很多重复功能的接口;我们没必要记住所有的接口,只需将核心接口记住并熟练使用就好啦;(如果遇到陌生的接口,直接查看文档就好了)。
先来看一下C++库里面string的构造函数。
1、string();
这是没有参数的构造函数(也就是默认构造函数),就是在创建string类对象时不需要传参数。
- #include
- #include
-
- using namespace std;
- void test()
- {
- string s1;
- }
-
- int main()
- {
- test();
- return 0;
- }
2、string(const char* s);
在创建string类对象时,传字符串(指针/字符数组),进行类对象的初始化。
使用:用字符串来初始化对象。
- void test()
- {
- string s1; //string();
- string s2("I Love You"); //string(const char* s);
- }
3、string(const char* s, size_t n);
与上面的构造函数不同的时,在传字符串指针的时,多一个参数n。
使用:用字符数组的前n个字符来初始化string类对象。
- void test()
- {
- string s1; //string();
- string s2("I Love You"); //string(const char* s);
- string s3("I Love You", 6); //string(const char* s, size_t n);
- }
可以通过调试来看对象s1,s2,s3的值
4、string(const string& str);
在创建string类对象,传同为string类类型的对象进行对象的初始化(也就是拷贝构造函数)。
使用:通过同类型的对象来初始化;
- void test()
- {
- string s1; //string();
- string s2("I Love You"); //string(const char* s);
- string s3("I Love You", 6); //string(const char* s, size_t n);
- string s4(s2); // string(const string& str);
- }
5、string(const string& str, size_t pos, size_t len=npos);
与拷贝构造不同的时,这个构造函数,还需要多传两个参数pos和len;
使用:用str中的第pos个位置后面len个字符进行初始化对象。
- void test()
- {
- string s1; //string();
- string s2("I Love You"); //string(const char* s);
- string s3("I Love You", 6); //string(const char* s, size_t n);
- string s4(s2); // string(const string& str);
- string s5(s2, 2, 4); //string(const string& str, size_t pos, size_t len=npos);
- }
6、string(size_t n, char ch);
创建string类对象,并将其内容设置成n个ch。
- void test()
- {
- string s1; //string();
- string s2("I Love You"); //string(const char* s);
- string s3("I Love You", 6); //string(const char* s, size_t n);
- string s4(s2); // string(const string& str);
- string s5(s2, 2, 4); //string(const string& str, size_t pos, size_t len=npos);
- string s6(6, 'H'); //string(size_t n, char ch);
- }
7、
C++文档中最后一个构造函数,涉及到迭代器,这里先不介绍,后面再使用。
析构函数没有函数重载,就不过多叙述。
1、string& operator= (const string& str);
使用:这个就是对于两个已经存在的string类类型对象,进行赋值操作。
- void test1()
- {
- string s1;
- string s2("Hello World");//拷贝构造
- s1 = s2;//赋值运算符重载
-
- }
2、string& operator= (const char* s);
使用:将一个字符串赋值给一个已经存在的string类类型对象。
- void test1()
- {
- string s1;
- string s2("Hello World");//拷贝构造
- s1 = s2;//赋值运算符重载
-
- string s3;
- s3 = "I Love ";//operator= (const char* s);
- }
3、string& operator= (char ch);
使用:将一个字符赋值给一个已经存在的string类类型对象。
- void test1()
- {
- string s1;
- string s2("Hello World");//拷贝构造
- s1 = s2;//赋值运算符重载
-
- string s3;
- s3 = "I Love ";//operator= (const char* s);
-
- string s4;
- s4 = 'c';//operator= (char ch);
- }
调试看一下赋值结果:
这里就看几个常用的(size、lengh、resize、capacity、clear和empty)。
size就是返回string对象中的字符(数据)个数(也就是字符串的长度)。
lengh和size一样,都是计算string对象中字符串的长度。
resize是设置string对象中字符串的内容,(如果n要小于size就将字符串数据个数设为n)如果n大于size,(空间不够就扩容)设置size后面的字符内容。如果给了参数c就设置成 c;如果没有给参数c,就默认设置成'\0'。
capacity 返回string对象中可用空间的大小(就像动态顺序表中capacity一样)。
clear是清理string中字符串的内容,让string对象中字符串变为空,并修改对象中size的值。
empty判断string对象是否为空;为空就返回true,否则返回false。
reserve是设置string的空间大小,如果传参n比capacity大就扩容;如果小,不一定会缩小容量。
- void test2()
- {
- string s1("I Love You !!!");
- //size 返回s1中字符串数据个数
- cout << "size = " << s1.size() << endl;
- //lengh 返回s1中字符串长度
- cout << "lengh = " << s1.length() << endl;
- //capacity 返回s1中可用空间大小
- cout << "capacity = " << s1.capacity() << endl;
- //设置s1中数据个数
- s1.resize(10);
- cout << s1 << endl;
- s1.resize(20, 'x');
- cout << s1 << endl;
- //清理s1中字符串内容,使其变成空
- s1.clear();
- cout << s1 << endl;
- //empty 判断s1中字符串是否为空
- cout << s1.empty() << endl;
- }
Element access 就是元素访问,(访问字符串中的数据)。
这里就主要看 operator[] 和at;
除了这样访问元素,还可以通过迭代器就行访问(下面在迭代器内容进行使用)。
operator[] 库里面实现了两个,一个用于普通对象的元素访问(可读可写),另一个就用于const修饰的对象的元素访问(只读)。
有了operator[] 这个运算符重载的函数,我们就可以像访问字符数组那样去通过下标访问string类型对象。
at这个函数是传递一个参数pos,获得string对象中字符串下标为pos的字符。(使用起来和operator[]一样)。
- void test3()
- {
- string s1("Hello world");
- for (int i = 0; i < s1.size(); i++)
- {
- s1[i]++; //普通对象调用char& operator[](size_t pos); 可以进行修改
- cout << s1[i] << " ";
- }
- cout << endl;
- const string s2("Hello World");
- for (int i = 0; i < s2.size(); i++)
- {
- //s2[i]++; //err
- //const修饰的对象调用const char& operator[](size_t pos) cosnt; 不能进行修改
- cout << s2[i] << " ";
- }
- cout << endl << endl;
- string s3("Hello world");
- for (int i = 0; i < s3.size(); i++)
- {
- s3.at(i)++; //普通对象调用char& at(size_t pos); 可以进行修改
- cout << s3.at(i) << " ";
- }
- cout << endl;
- const string s4("Hello World");
- for (int i = 0; i < s4.size(); i++)
- {
- //s4.at[i]++; //err
- //const修饰的对象调用const char& at(size_t pos) cosnt; 不能进行修改
- cout << s4.at(i) << " ";
- }
- cout << endl;
-
- }
Modifiers ,库里面提供了以下这些函数,对string对象中字符串进行设置(修改)
库里面实现了三个+=运算符重载函数,我们在使用时就可以像内置类型那样+=;
这里可以+=一个string类类型对象、字符串(字符数组)或者一个字符。
- void test4()
- {
- string s1("hello ");
- string s2("world ");
- s1 += s2;
- cout << s1 << endl;
-
- char str[] = "ever";
- s1 += str;
- cout << s1 << endl;
-
- s1 += 'o';
- s1 += 'n';
- s1 += 'e';
- cout << s1 << endl;
- }
库里面实现的append感觉有点冗余;
append可以增加string类型对象(也可以是其中的一部分)、字符串(可以是其中的一部分)、n个字符;传参也可以是迭代器。
- void test5()
- {
- string s1("hello ");
- string s2("world ");
- //s1.append(s2); 也可以直接添加s2的全部
- s1.append(s2, 1, 3);
- //从下标为1的位置开始,添加后面3个字符
- cout << s1 << endl;
-
- s1.append("everone");
- //s1.append("everone",4);
- //也可以这样,只添加前4个字符
- cout << s1 << endl;
-
- s1.append(3, 'x');
- //添加3个字符 'x'
- cout << s1 << endl;
- }
push_back就是在string对象后面添加一个字符。(比较简单就不测试了)。
assgin是设置string类对象的内容,(可以用同类型对象来设置(也可以是一部分)、也可以用字符串设置(或者其中一部分)、也可以将strng对象内容设置成n个字符;(也可以传迭代器))。
- void test5()
- {
- string s1("hello ");
- string s2("world ");
- //s1.append(s2);
- s1.assign(s2, 1, 3);
- cout << s1 << endl;
-
- s1.assign("everone");
- //s1.append("everone",4);
- cout << s1 << endl;
-
- s1.assign(3, 'x');
- cout << s1 << endl;
- }
insert与assgin不同,要比assgin多一个参数,这个参数pos就决定了从string对象中字符串哪个位置开始设置字符串的内容。
- void test5()
- {
- string s1("hello ");
- string s2("world ");
- //s1.append(s2);
- s1.insert(3, s2, 1, 3);
- cout << s1 << endl;
-
- s1.insert(8, "everone");
- //s1.append("everone",4);
- cout << s1 << endl;
-
- s1.insert(15, 3, 'x');
- cout << s1 << endl;
- }
erase是删除string对象中字符串的数据,(删除pos位置后面的len和字符);如果不传参就是全部删除(如果只是不传len的参数,就默认删除pos后面所有的数据)。
- void test6()
- {
- string s1("hello world !!!");
- cout << s1 << endl;
- s1.erase(11, 3);
- cout << s1 << endl;
- s1.erase(5);
- cout << s1 << endl;
- s1.erase();
- cout << s1 << endl;
- }
replace 替换string对象字符串中的数据;
函数的前两个参数,就是指需要替换的位置(pos位置后面的len个字符)。
可以替换成string类类型对象(也可以是一部分)、字符串(也可以是一部分)、n个字符或者迭代器指向的区间。
- void test7()
- {
- string s1("Hello World ");
- string s2("everone");
- cout << s1 << endl;
- s1.replace(6, 5, s2);
- cout << s1 << endl;
-
- s1.replace(6, 5, "I Love");
- cout << s1 << endl;
-
- s1.replace(6, 5, 3, '*');
- cout << s1 << endl;
- }
swap是库里面实现的一个函数(交换两个string类类型对象)。
- void test8()
- {
- string s1("hello world");
- string s2("I Love you");
- cout << "s1: " << s1 << endl;
- cout << "s2: " << s2 << endl;
- s1.swap(s2);
- cout << "s1: " << s1 << endl;
- cout << "s2: " << s2 << endl;
- }
字符串操作,主要操作有返回字符串指针、查找、拷贝、获得子串等。(这里就使用其中的一部分)。
c_str返回string对象里的字符串指针。(需要注意的就是返回的指针是const修饰的,必须用const指针接受)。
- void test9()
- {
- string s1("Hello World");
- const char* s = s1.c_str();
- cout << s << endl;
- }
find查找, 在string对象里查找指定字符(string类类型对象、字符串、字符),也可以指定区间进行查找。找到就返回下标;没有找到返回npos。
- void test9()
- {
- string s1("Hello World");
- string s2("Hello");
- size_t f1 = s1.find(s2);
- size_t f2 = s1.find("Love");
- size_t f3 = s1.find('o');
-
- cout << f1 << endl;
- cout << f2 << endl;
- cout << f3 << endl;
- }
substr获得string对象字符串中的子串。这结合上面的find,
实现将一个网址的协议、域名以及资源分开。
- void test10()
- {
- string s1("https://blog.csdn.net/LH__1314?type=blog");
- size_t f1 = s1.find("://", 0);
- string protocol = s1.substr(0, f1);
-
- size_t f2, f3;
- f2 = s1.find('/', f1 + 3);
- string domain = s1.substr(f1 + 3, f2 - (f1 + 3));
- string resource = s1.substr(f2 + 1);
-
- cout << protocol.c_str() << endl;
- cout << domain.c_str() << endl;
- cout << resource.c_str() << endl;
- }
inerator就是所谓的迭代器,在string中迭代器作用没有那么大,(甚至可以将它理解为指针(但迭代器不是指针))。
在使用这些函数之前,要先知道一种类型,迭代器 iterator
string::iterator it;
因为在其他容器里也存在迭代器,所以定义时要指定类域。
这里的两个函数返回的都是迭代器,begin返回的是起始位置,end返回的是终止位置。
这里库里面还实现了const修饰的函数,const修饰的迭代器类型有所不同
string::const_iterator it;
这里直接使用:
- void test11()
- {
- string s1("I Love You");
- string::iterator it = s1.begin();
- while (it != s1.end())
- {
- cout << *it << " ";
- it++;
- }
- cout << endl;
- }
有了迭代器,我们就可以实现一种语法 范围for
- void test11()
- {
- string s1("I Love You");
- for (auto ch : s1)
- {
- cout << ch << " ";
- }
- }
到这里,string类的基本使用就结束了。
感谢各位大佬支持并指出问题,
如果本篇内容对你有帮助,可以一键三连支持以下,感谢支持!!!