data:image/s3,"s3://crabby-images/c55d2/c55d2c4814e82eda157177a87417b907cd488b01" alt=""
我们接着【C++】string类(接口使用详解 上)-CSDN博客 继续介绍string的使用。
1.string类对象的修改操作
data:image/s3,"s3://crabby-images/37235/3723594038f501e06ad29a9409a6566dd1605e8e" alt=""
我们就说一下用的比较多的接口。
1.1 operator+=
data:image/s3,"s3://crabby-images/94165/9416590fee906b4832f22629837089c30e941094" alt=""
这个接口可以尾插一个字符,或者一个字符串,或者一个对象。
string s1("hello world!");
string s2("qqqqqqq");
s1 += 'x'; // 尾插一个字符
s1 += "yyyyy"; //尾插一个字符串
s1 += s2; //尾插一个对象
cout << s1 << endl;
data:image/s3,"s3://crabby-images/48b9c/48b9c3a221570e42cdc8f8bb4ab5ae6a07099875" alt=""
在实践中直接用这个接口尾插就行。
1.2 append和 push_back
这两个接口也是尾插的,了解一下即可,我们一般就用前面介绍的operator+=。
push_back是尾插一个字符。
data:image/s3,"s3://crabby-images/204b5/204b50705cd85a83a8ae72333f29d45aefccc5f5" alt=""
append是尾插一个字符串,对象,对象的一部分等等,详细的看文档。
data:image/s3,"s3://crabby-images/14dbf/14dbf99742cfbb1e820c3c0c3ebfd31390c746ab" alt=""
string s1("hello world!");
s1.push_back('q'); //尾插一个字符
s1.append("xxxxxx"); //尾插字符串
data:image/s3,"s3://crabby-images/91552/9155214a3d3a66daa751c1170cc6bf53aa84c11b" alt=""
常用的用法就是上面这样,append的其他接口基本不咋用。
1.3 insert
这是插入数据的接口,提供了7个,不过大部分用的不多。
data:image/s3,"s3://crabby-images/8c51f/8c51f1432fd04456d332b440466a54cf5850353c" alt=""
常用的用法就两个,用这个头插,或者字符串中间位置插入数据。其他接口用的不多。
string s1("hello world!");
cout << s1 << endl;
s1.insert(0, 2, 'a'); //在下标为0的位置开始插入2个a
cout << s1 << endl;
data:image/s3,"s3://crabby-images/94c0b/94c0b9ed8b8816fe55b584b2805297a3f9e7ee33" alt=""
string s1("hello world!");
cout << s1 << endl;
s1.insert(0, 2, 'a');
cout << s1 << endl;
s1.insert(8, 4, 'x');
cout << s1 << endl;
data:image/s3,"s3://crabby-images/cd081/cd081f28dcceb3c67e16c693bcf6ffc4da1893e7" alt=""
但是头插和中间插入的使用需谨慎,我们学过顺序表可以知道,头插或者中间插入需要把后面的数据都往后移动,如果空间不够还要扩容。
在使用这些接口的时候,不确定用法就看文档介绍。
1.4 erase
erase是用来删除数据的。
data:image/s3,"s3://crabby-images/ca4b5/ca4b56beccbb15b2a396dc6a64067d5c3ba2549f" alt=""
在实践中用的最多的还是第一个接口:从pos的位置开始删除npos个数据。
string s1("hello world");
s1.erase(6, 1); //从下标为6的位置,删除1个数据
cout << s1 << endl;
data:image/s3,"s3://crabby-images/43961/439617e6b9738709b54b6c0ca76de493193117a2" alt=""
如果我们要头删,如下。
s1.erase(0, 1); //头删
cout << s1 << endl;
data:image/s3,"s3://crabby-images/5c587/5c58723518275fbbe262701222ace3b18833face" alt=""
这个接口的npos也是缺省参数,pos位置后面字符剩多少就是多少 。
如果我们想把pos位置之后的全删了,第二个参数可以不传,就用npos的缺省值。
string s1("hello world");
s1.erase(6); //从6的位置开始后面全删除
cout << s1 << endl;
data:image/s3,"s3://crabby-images/e3bf9/e3bf9c8872f3b25e182c61e04fdaa50baceb8694" alt=""
还提供了一个迭代器的版本,就是(3).
data:image/s3,"s3://crabby-images/040ad/040ad5970cda2f5b7421681edad6186d465fcb5e" alt=""
用迭代器版本头删也可以。
s1.erase(s1.begin()); //迭代器版头删
cout << s1 << endl;
data:image/s3,"s3://crabby-images/3cdf0/3cdf0998890e8aba8c6f17547459ceb6c946f616" alt=""
用迭代器版本尾删也可以。
s1.erase(--s1.end()); //迭代器版尾删
cout << s1 << endl;
data:image/s3,"s3://crabby-images/ca77a/ca77a1e2d4973aaf133a1a8fcd674e37b2e84e8e" alt=""
注意 括号里是s1.end()的前置减减,因为end()返回的是最后一个字符的下一位,减减end()才是返回最后一个字符。
//迭代器版尾删另一种写法
s1.erase(s1.size()-1, 1);
1.5 replace
replace就是替换,它提供的接口也是非常多。
data:image/s3,"s3://crabby-images/4373c/4373c08265c11031f47e120284303ed9738ee5bb" alt=""
大概就是我们可以把pos位置开始的len个字符替换字符串、字符串的一部分,对象,对象的一部分,迭代器,迭代器的一部分。来简单运用一下。
少被替换多
string s1("hello world");
//把5位置开始的1个字符替换成2个%
s1.replace(5, 1, "%%");
cout << s1 << endl;
data:image/s3,"s3://crabby-images/20c36/20c3682325fde0ab9d72565e6025cb86beebe230" alt=""
多被替换少
string s1("hello world");
s1.replace(5, 1, "%%");
cout << s1 << endl;
//把0开始的4个字符替换成3个x
s1.replace(0, 4, "xxx");
cout << s1 << endl;
data:image/s3,"s3://crabby-images/1787a/1787aa48f2a15621f2a20abc4fb9d41886b45950" alt=""
等大小替换
s1.replace(0, 3, "yyy");
cout << s1 << endl;
data:image/s3,"s3://crabby-images/574c3/574c39899a783fc1548324796b48a5ece605a67f" alt=""
1.6 swap和pop_back
一个是交换,一个是尾删,了解一下即可。
data:image/s3,"s3://crabby-images/7170a/7170aa9b026ff4f9fec3157478095e15622c1e56" alt=""
string的其他接口在这就不一一介绍了,大家在使用的时候不清楚的话查一下文档就好了。
data:image/s3,"s3://crabby-images/88bc7/88bc7adbfb8558d1019fdfab5b23a95246feeee4" alt=""
2.find系列接口
2.1 find
从pos位置查找字符,字符串,对象,返回值是size_t类型。
data:image/s3,"s3://crabby-images/cdcbe/cdcbe3fac2785b5a814e79ad730334e9fb5e8650" alt=""
如果找到了,返回找到的第一个目标的下标位置。 如果没找到,返回npos。
举个例子。
string s2("hello world hello csdn");
size_t pos = s2.find(" "); //找空格
如果我们找到空格,把空格全部换成%%。
size_t pos = s2.find(" "); //找空格,把下标存在pos里
while (pos != string::npos)
{
s2.replace(pos, 1, "%%"); //替换
pos = s2.find(" "); //找下一个空格
}
cout << s2 << endl;
data:image/s3,"s3://crabby-images/3dd14/3dd143d99fa9e3e94557407409e03f4acd07d7b6" alt=""
但是这样写的话,我们不给第二个参数传参,find每次都要从头找,很麻烦,我们给第二个参数传参,可以从指定位置开始找。把while循环里面的第二条语句改一下。
while (pos != string::npos)
{
s2.replace(pos, 1, "%%"); //替换
pos = s2.find(" ", pos + 2); //pos+2位置开始找下一个空格
}
这里第二个参数是pos+2,是因为我们把空格这一个字符替换成了两个%。结果是一样的。
2.2 rfind 和substr
data:image/s3,"s3://crabby-images/e79cf/e79cfbc120a4330760cb38f2b6246ac00a907527" alt=""
rfind和find差不多,只不过rfind是倒着找。
data:image/s3,"s3://crabby-images/e2ace/e2ace20c99126a9b679188f07d1d8fe52dd6242c" alt=""
substr是把pos位置的len个字符拿出来拷贝到新的string里,这里npos是老朋友了,如果npos不给值,就是拷贝到结尾。返回值类型是string。
data:image/s3,"s3://crabby-images/71691/71691078e0456855892ca3420d5b5a7839221747" alt=""
rfind和substr结合起来用,可以用在找文件的后缀。
//假设文件是string.cpp
string s3("string.cpp");
size_t pos = s3.rfind('.'); //从后开始找.
string suffix = s3.substr(pos); //找到后把后缀存到suffix里
cout << suffix << endl;
data:image/s3,"s3://crabby-images/e7b6f/e7b6f3db6bc58cb3627f86e270786487e5fc803a" alt=""
//假设文件是string.cpp.zip
string s3("string.cpp.zip");
size_t pos = s3.rfind('.'); //从后开始找.
string suffix = s3.substr(pos); //找到后把后缀存到suffix里
cout << suffix << endl;
data:image/s3,"s3://crabby-images/a851e/a851e29e7e12977395d908587fa4e98486441c16" alt=""
2.3 find_first_of
名字看起来是找第一个,其实并不是,别被名字迷惑哦。
data:image/s3,"s3://crabby-images/81fdc/81fdceba7839937db244dc9b87b082c9a230961a" alt=""
data:image/s3,"s3://crabby-images/37bac/37bac72b90724c2832734c1d44e60b6cffe7199b" alt=""
find_first_of 是找任意一个字符,举个例子,如下。
string str ("Please, replace the vowels in this sentence by asterisks.");
cout << str << '\n';
size_t found = str.find_first_of("abcd");//找abcd的任意一个,a或者b或者c或者d
while (found!=string::npos)
{
str[found]='*'; //只要是a或者b或者c或者d都会被替换成*
found=str.find_first_of("abcd",found+1);
}
cout << str << '\n';
data:image/s3,"s3://crabby-images/f25e5/f25e5f209f69368b898d7d0749388f6f07ffcbf9" alt=""
大家一定不能被名字迷惑!是传的任意一个字符都会被找到!
从下面这句代码也可以看出,第二个参数是从pos位置开始找,不传默认为0位置开始。
found=str.find_first_of("abcd",found+1);
2.4 find_last_of
这个函数和find_first_of 功能是一样的,只不过是从后面往前找。
data:image/s3,"s3://crabby-images/a3d03/a3d03bbf56e759adecb01584dccc928bf9b2b945" alt=""
data:image/s3,"s3://crabby-images/602e3/602e3df79b716620b64619f90f62e8ea11c8127e" alt=""
举个例子,还是用文件举例。假设我们要把文件路径和文件名分隔开看,Windows和Linux下的文件分隔符不一样,Windows下是\,Linux下是/,代码如下。
void SplitFilename (const std::string& str)
{
cout << "Splitting: " << str << '\n';
size_t found = str.find_last_of("/\\");//从后往前,找/和\\的任意一个
cout << " path: " << str.substr(0,found) << '\n';
cout << " file: " << str.substr(found+1) << '\n';
}
int main ()
{
string str1 ("/usr/bin/man"); //Linux路径
string str2 ("c:\\windows\\winhelp.exe");//Windows路径
SplitFilename (str1);
SplitFilename (str2);
return 0;
}
data:image/s3,"s3://crabby-images/98360/9836039fe3a8bbe6371f1c5260357d7fd3babca6" alt=""
2.5 find_first_not_of和find_last_not_of
这两个和find_first_of、find_last_of相反,比如说前面我们要把是abcd任意一个的找到,这两个接口就是把不是abcd任意一个都找到。
data:image/s3,"s3://crabby-images/b20e0/b20e0e36244c86f55fc2b5ade3d155cd5140d8fb" alt=""
那我们还是拿前面的abcd举例吧,这次换成abcdef。
string str ("Please, replace the vowels in this sentence by asterisks.");
cout << str << '\n';
size_t found = str.find_first_not_of("abcdef");//找不是abcdef的任意一个
while (found!=string::npos)
{
str[found]='*'; //除了abcdef,其它都会被替换成*
found=str.find_first_of("abcdef",found+1);
}
cout << str << '\n';
除了abcdef,其它都被替换成*了。
find_last_not_of是一样的,只是从后往前找。
data:image/s3,"s3://crabby-images/e7d5c/e7d5c18321d6db907d86076a893b28845ff70ea7" alt=""
3.string类的非成员函数
data:image/s3,"s3://crabby-images/6523e/6523efa922186653a14b7c04ba080e1d00e36b0a" alt=""
3.1 operator+
data:image/s3,"s3://crabby-images/20420/20420a8fdaf61cc6ce0794207d8935b9be678145" alt=""
这个函数为什么没有写成成员函数,而是重载成全局的呢?因为它主要想支持字符串+string的功能。如下。
string s4("hello");
string s5 = s4 + "world";//string+字符串
cout << s5 << endl;
string s6 = "world" + s4;//字符串+string
cout << s6 << endl;
data:image/s3,"s3://crabby-images/1f1b9/1f1b96d9c43b636eb0013f366a6b73a495d44b68" alt=""
我们说过重载运算符,二元的,两个操作数左右顺序一一对应,如果是成员函数,左操作数只能是string,重载成全局的函数,就可以支持字符串是左操作数。
3.2 比较大小
data:image/s3,"s3://crabby-images/dda94/dda945449a066ccdf0bde770f484a94f69445666" alt=""
比较大小的接口也是重载了一大堆,在这里就不说了 。
3.3 <<、>>、 swap
data:image/s3,"s3://crabby-images/e71d0/e71d02f1817b0527cc8de8047c724747a07d43f1" alt=""
swap留到后面再说,流插入和流提取也没啥说的~
3.4 getline
cin和scanf默认我们输入空格或者换行符就停止一个输入,如果我们要输入hello world,输入到str里,str只会存hello进去,world会认为是另一个对象的。
data:image/s3,"s3://crabby-images/8d724/8d724d411844c75372e5206f19debbd42197ebb6" alt=""
我们用getline输入的话,第一个参数传cin,第二个参数传对象,第三个参数不传,默认以换行符未结束标志。
data:image/s3,"s3://crabby-images/220dd/220ddc4baf47837b11a736590ebd33e6f39a6701" alt=""
第三个参数传的话,传什么,就以什么为结束标志。
data:image/s3,"s3://crabby-images/d493a/d493a06ed99b683e2acca9fe841870a631b2646e" alt=""
string的其他接口在这就不一一介绍了,大家在使用的时候不清楚的话查一下文档就好了。
到这string的使用就全部说完了,拜拜~
data:image/s3,"s3://crabby-images/be497/be4976a7a8bd6c9aad2c6e197b15522a2e329138" alt=""