目录
string类的常用接口说明
string类对象的容量操作
size & max_size & length & capacity
empty & clear
reserve & resize
string类对象的元素访问
at & back & front
string类对象的修改操作(字符串操作)
substr & find & rfind
find_first_of & find_last_of
find_first_not_of & find_last_not_of
compare
string类的常用接口说明
string类对象的容量操作
函数名称 功能说明 size(重点) 返回字符串有效字符长度(不包括\0) length 返回字符串有效字符长度 capacity 返回空间总大小 empty (重点) 检测字符串释放为空串,是返回true,否则返回false clear (重点) 清空有效字符 reserve (重点) 为字符串预留空间** resize (重点) 将有效字符的个数该成n个,多出的空间用字符c填充
size & max_size & length & capacity
void test_string7()
{string s1 = "hello world";cout << s1.size() << endl; //输出11cout << s1.length() << endl; //输出11cout << s1.capacity() << endl; //输出15string s2;cout << s2.size() << endl; //输出0cout << s2.length() << endl; //输出0cout << s2.capacity() << endl; //输出15
}
这三个相当简单,不过多赘述,注意的是size返回字符串有效字符长度(不包括\0)
size()与length()方法底层实现原理完全相同,引入size()的原因是为了与其他容器的接口保持一 致,一般情况下基本都是用size()。
max_size返回字符串可以达到的最大长度,没啥用处,不用深度学习。
empty & clear
void test_string8()
{string s1 = "hello world";if (s1.empty())//输出nocout << "yes" << endl;elsecout << "no" << endl;s1.clear();if (s1.empty())//输出yescout << "yes" << endl;elsecout << "no" << endl;
}
empty函数判度字符串是否为空,clear函数清空字符串。
clear()只是将string中有效字符清空,不改变底层空间大小。
reserve & resize
在了解这两个函数之前先了解编译器是怎么存储string类的数据的:
vs下string的结构
- string总共占28个字节,内部结构稍微复杂一点,先是有一个联合体,联合体用来定义string中字 符串的存储空间:
- 当字符串长度小于16时,使用内部固定的字符数组来存放 ;当字符串长度大于等于16时,从堆上开辟空间。
这种设计也是有一定道理的,大多数情况下字符串的长度都小于16,那string对象创建好之后,内 部已经有了16个字符数组的固定空间,不需要通过堆创建,效率高。
g++下string的结构
- G++下,string是通过写时拷贝实现的,string对象总共占4个字节,内部只包含了一个指针,该指针将来指向一块堆空间 。
reserve(size_t res_arg=0):为string预留空间,不改变有效元素个数,当reserve的参数小于 string的底层空间总大小时,reserve不会改变容量大小。 再三强调reserve和reverse的拼写区别!!
void TestPushBack()
{string s;// 知道需要多少空间,提前开好s.reserve(200);//s[100] = 'x'; //无法赋值
}
注意:这里的开辟了200个空间只是提前预留好的,不能进去访问,否则会出错
void test_string9()
{string s1;// reserve 保留// reverse 逆置 反转s1.reserve(100);cout << s1.capacity() << endl;s1.reserve(20);cout << s1.capacity() << endl;
}
另外注意不同编译器下reserve的处理方式不同,可能会缩容也可能不会。
如果要缩容,也只是另外开辟一半的空间,将原先空间的值拷贝一半进去,不能够在原来空间上缩容,归根结底还是空间换时间。
resize(size_t n) 与 resize(size_t n, char c)都是将字符串中有效字符个数改变到n个,不同的是当字符个数增多时:resize(n)用0来填充多出的元素空间,resize(size_t n, char c)用字符c来填充多出的元素空间。
void test_string11()
{string s1;//s1.resize(5, '0');s1.resize(5);s1[4] = '3';s1[3] = '4';s1[2] = '5';s1[1] = '6';s1[0] = '7';cout << s1 << endl;// 76543// 插入(空间不够会扩容)string s2("hello world");s2.resize(20, 'x');cout << s2 << endl;//hello worldxxxxxxxxx// 删除s2.resize(5);cout << s2 << endl;//hello
}
注意:resize在改变元素个数时,如果是将元素个数增多,可能会改变底层容量的大小,如果是将元素个数减少,底层空间总大小不变。
string类对象的元素访问
operator [ ]在上文已经详细解释,不再过多描述。
at & back & front
at 能获取字符串中的字符,返回对字符串中位置 pos 处的字符的引用。
void test_string12()
{string s2("hi");// 该函数会自动检查 pos 是否是字符串中字符的有效位置// (即 pos 是否小于字符串长度)// 如果不是,则抛出out_of_range异常。try{s2.at(10);//非法访问}catch (const exception& e){cout << e.what() << endl;//输出 invalid string position}
}
front 访问第一个字符,back 访问最后一个字符
void test_string13()
{string s = "hello world";s.back() = 'D';s.front() = 'H';cout << s << endl;//输出Hello worlD
}
string类对象的修改操作(字符串操作)
substr & find & rfind
void test_string14()
{string file("string.cpp.zip");// 在字符串中搜索其参数指定的序列的第一次出现项size_t pos1 = file.find('.');// 在字符串中搜索其参数指定的序列的最后一次出现项size_t pos2 = file.rfind('.');// 子字符串是对象中从字符位置开始并跨越字符(或直到字符串末尾,以先到者为准)的部分// string suffix = file.substr(pos, file.size() - pos);string suffix1 = file.substr(pos1); //默认到结束cout << suffix1 << endl;//输出 .cpp.zipstring suffix2 = file.substr(pos2);cout << suffix2 << endl;//输出 .zipstring suffix3 = file.substr(pos1,4);//从pos1开始跨越四个字符cout << suffix3 << endl;//输出 .cpp
}
find_first_of & find_last_of
void test_string15()
{// strtokstd::string str("Please, replace the vowels in this sentence by asterisks.");std::size_t found1 = str.find_first_of("aeiou");//在字符串中搜索与其参数中指定的任何字符匹配的第一个字符std::size_t found2 = str.find_last_of("aeiou");//在字符串中搜索与其参数中指定的任何字符匹配的最后一个字符str[found1] = '*';str[found2] = '*';std::cout << str << '\n';//输出 Pl*ase, replace the vowels in this sentence by aster*sks.
}
find_first_not_of & find_last_not_of
void test_string16()
{// strtokstd::string str("Please, replace the vowels in this sentence by asterisks.");std::size_t found1 = str.find_first_not_of("aeiou");//在字符串中搜索与其参数中指定的任何字符匹配的第一个字符std::size_t found2 = str.find_last_not_of("aeiou");//在字符串中搜索与其参数中指定的任何字符匹配的最后一个字符str[found1] = '*';str[found2] = '*';std::cout << str << '\n';//输出 *lease, replace the vowels in this sentence by asterisks*
}
compare
void test_string17()
{string str1("green apple");string str2("red apple");if (str1.compare(str2) != 0)std::cout << str1 << " is not " << str2 << '\n';// 输出 green apple is not red apple
}
C++98设置的类型过多,个人绝对记忆太过于繁琐,只需要了解几个有用的即可,比较是按照字典序大小比较的(ascii码比较)