string
C++STL中加入string类型,对字符串常用需求功能进行了封装,使得操作起来更方便,且不易出错。如果要使用string,需要添加string头文件,即#include<string>,注意string.h和string是不一样的头文件。除此之外要想使用string,还要在头文件下面加上一句“using namespace std”,这样就可以在代码中使用string了。
string的定义:
定义方式与基本数据类型相同,只需要在string后面跟上变量名称即可。
eg. string str;如果需要初始化,可以直接给string类型的变量赋值,string str = “hello”。
String中内容的访问:
(1) 通过下标访问,可以直接像字符数组那样去访问string
string str = "hello" ;
for(int i=0;i<str.size();i++){
printf("%c",str[i]) ;
}
输出结果:hello
如果用string读入和输出字符串一般只能用cin和cout,使用printf输出也是可以的,但是要先将string装换为字符数组进行输出
string str ;
cin >> str ;
cout << str << endl ;
printf("%s\n",str.c_str()) ;
return 0 ;
输入:hello
输出:hello
hello
(2) 通过迭代器访问,string的访问一般采用第一种方式,但是string有很多函数要求以迭代器为参数。string迭代器的定义方式为string::iterator it,可以使用*it来访问string中的每一位。String迭代器的使用方法与其他容器的迭代器的使用方法类似,string和vector一样,支持直接对迭代器进行加减某个数字。
示例:
string str = "hello" ;
string::iterator it ;
for(it = str.begin();it!=str.end();it++){
cout << *it ;
}
it = str.begin() ;
cout << endl ;
cout << *(it+1) << "<--->" << str[1] << endl ;
结果: hello
e<--->e
string常用函数:
string提供的函数有很多,这里紧介绍常用的string函数
(1) operator+=拼接
这是string的加法,可以将两个string直接拼接起来。
string str1 = "hello" ;
string str2 = " world" ;
cout << str1 + str2 << endl ;
结果:hello world
(2) compare operator比较
两个string类型可以直接使用==,!=,<,<=,>,>=比较大小,比较规则是字典序。
string str1 = "aa" ;
string str2 = "ab" ;
string str3 = "abc" ;
if(str1 != str2){
cout << "not same" << endl ;
}else{
cout << "same" << endl ;
}
if(str3>str2){
cout << "str3>str2" << endl ;
}
结果:not same
str3>str2
(3) length()/size()取得大小
length()返回string的长度,即string存放的字符数,时间复杂度O(1)。size()和length()基本相同。
string str1 = "hello" ;
cout << str1.size() << ' ' << str1.length() << endl ;
结果:5 5
(4) insert()插入(原字符串不会被覆盖)
insert()函数有很多种写法,这里列出几个常用的写法,时间复杂度度O(N)。
insert(pos,string),在pos号位置插入string。
insert(it,it1,it2),it为原字符串欲插入的位置,it2和it3为待插字符串的首尾迭代器,用来表示串[it1,it2)将被插在it的位置上。
string str1 = "hello world" ;
string str2 = "Maric" ;
str1.insert(str1.begin()+5,str2.begin(),str2.end()) ;
cout << str1 << endl ;
str2.insert(5,"hello") ;
cout << str2 << endl ;
结果:helloMaric world
Marichello
(5) erase()删除
erase()有两种用法,删除单个元素,上出一个区间内所有元素。时间复杂度O(N)。
a. erase(it)用于删除单个元素,it为需要删除的元素的迭代器。
b. 删除一个区间的元素有两种方法:
第一种是erase(st,ed),st,ed为string迭代器,表示删除区间[st,ed)之间的元素。
第二种是erase(pos,len),其中pos为需要删除的起始位置,len为删除的字符个数。
string str1 = "hello world" ;
str1.erase(str1.begin()) ;
cout << str1 << endl ;
str1.erase(str1.begin()+5,str1.end()) ;
cout << str1 << endl ;
str1.erase(1,2) ;
cout << str1 << endl ;
结果:ello world
ello
eo
(6) clear()清空
clear()函数用来清空string中的数据,时间复杂度O(1)。
string str1 = "hello world" ;
cout << str1.length() << endl ;
str1.clear() ;
cout << str1.size() << endl ;
结果:11
0
(7) substr()截取子串
substr(pos,len)返回的是以pos位开始长度为len的子串,时间复杂度O(len)。
string str1 = "hello world" ;
cout << str1.substr(6,5) << endl ;
结果:world
(8) string::npos
npos 是一个常数,用来表示不存在的位置,类型一般是std::container_type::size_type 许多容器都提供这个东西。取值由具体实现决定,一般是-1,但是由于是unsigned_int类型,也可以认为是unsigned_int类型的最大值,这样做,就不会存在移植的问题了。
不建议将find的返回值作为是否匹配的依据。最好使用示例的使用方式。npos主要是和find()函数搭配使用,用以作为find()函数失配时的返回值,举个例子具体解释一下:
if (a.find(b) != string::npos) {
cout << "Yes!" << endl;
} else {
cout << "No!" << endl;
}
(9) find()查找
str.find(str1),当str1是str的子串时,返回其在str中第一次出现的位置。如果str1不是str的子串,那么返回string::npos
str.find(str1,pos),从str的pos号位开始匹配str1,返回值与上面的相同,时间复杂度为O(nm),其中n和m分别为str和str1的长度。
string str1 = "hello world" ;
string str2 = "world" ;
if(str1.find(str2) != string::npos){
cout << str1.find(str2) << endl ;
cout << str1.find(str2,3) << endl ;
}else{
cout << "match failure" << endl ;
}
结果:6
6
小结:find函数的返回值是整数,假如字符串存在包含关系,其返回值必定不等于npos,但如果字符串不存在包含关系,那么返回值就一定是npos。
(10) replace()
str.replace(pos,len,str1),把str从pos号位开始,长度为len的子串替换为str1。
str.replace(it1,it2,str1)把str的迭代器[it1,it2)替换为str1
string str1 = "hello world" ;
string str2 = "kangkang" ;
cout << str1.replace(6,5,str2) << endl ;
cout << str1.replace(str1.begin()+6,str1.end(),str2) << endl ;
结果:hello kangkang
hello kangkang
666666 牛啊