C++
提供了以下两种类型的字符串表示形式:
C
风格字符串C++
引入的string
类类型
C 风格字符串
C 风格的字符串起源于 C 语言,并在 C++ 中继续得到支持。
字符串实际上是使用 null
字符'\0'
终止的一维字符数组。因此,一个以 null 结尾的字符串,包含了组成字符串的字符。
下面的声明和初始化创建了一个 "Hello"
字符串。由于在数组的末尾存储了空字符,所以字符数组的大小比单词 "Hello"
的字符数多一个。
char greeting[6] = {'H', 'e', 'l', 'l', 'o', '\0'};
依据数组初始化规则,上面的语句写成以下语句:
char greeting[] = "Hello";
以下是 C/C++ 中定义的字符串的内存表示:
C++ 编译器会在初始化数组时,自动把 ‘\0’ 放在字符串的末尾。让我们尝试输出上面的字符串:
实例
#include <iostream>
using namespace std;
int main ()
{
char greeting[6] = {'H', 'e', 'l', 'l', 'o', '\0'};
cout << "Greeting message: ";
cout << greeting << endl;
return 0;
}
当上面的代码被编译和执行时,它会产生下列结果:
Greeting message: Hello
C++ 中有大量的函数用来操作以 null 结尾的字符串
函数 | 目的 |
---|---|
strcpy(s1, s2) | 复制字符串 s2 到字符串 s1。 |
strcat(s1, s2) | 连接字符串 s2 到字符串 s1 的末尾。 |
strlen(s1) | 返回字符串 s1 的长度。 |
strcmp(s1, s2) | 如果 s1 和 s2 是相同的,则返回 0;如果 s1<s2 则返回值小于 0;如果 s1>s2 则返回值大于 0。 |
strchr(s1, ch) | 返回一个指针,指向字符串 s1 中字符 ch 的第一次出现的位置。 |
strstr(s1, s2) | 返回一个指针,指向字符串 s1 中字符串 s2 的第一次出现的位置。 |
下面的实例使用了上述的一些函数:
实例
#include <iostream>
#include <cstring>
using namespace std;
int main ()
{
char str1[11] = "Hello";
char str2[11] = "World";
char str3[11];
int len ;
// 复制 str1 到 str3
strcpy( str3, str1);
cout << "strcpy( str3, str1) : " << str3 << endl;
// 连接 str1 和 str2
strcat( str1, str2);
cout << "strcat( str1, str2): " << str1 << endl;
// 连接后,str1 的总长度
len = strlen(str1);
cout << "strlen(str1) : " << len << endl;
return 0;
}
当上面的代码被编译和执行时,它会产生下列结果:
strcpy( str3, str1) : Hello
strcat( str1, str2): HelloWorld
strlen(str1) : 10
C++ 中的 String 类
C++
标准库提供了 string 类类型,支持上述所有的操作,另外还增加了其他更多的功能。先来看看下面这个实例:
实例
#include <iostream>
#include <string>
using namespace std;
int main ()
{
string str1 = "Hello";
string str2 = "World";
string str3;
int len ;
// 复制 str1 到 str3
str3 = str1;
cout << "str3 : " << str3 << endl;
// 连接 str1 和 str2
str3 = str1 + str2;
cout << "str1 + str2 : " << str3 << endl;
// 连接后,str3 的总长度
len = str3.size();
cout << "str3.size() : " << len << endl;
return 0;
}
当上面的代码被编译和执行时,它会产生下列结果:
str3 : Hello
str1 + str2 : HelloWorld
str3.size() : 10
string类提供了一系列针对字符串的操作,比如:
append()
– 在字符串的末尾添加字符find()
– 在字符串中查找字符串insert()
– 插入字符
5.length()
– 返回字符串的长度replace()
– 替换字符串substr()
– 返回某个子字符串
下面是关于string类的实例:
#include <iostream>
#include <string>
using namespace std;
int main()
{
//定义一个string类对象
string http = "www.runoob.com";
//打印字符串长度
cout<<http.length()<<endl;
//拼接
http.append("/C++");
cout<<http<<endl; //打印结果为:www.runoob.com/C++
//删除
int pos = http.find("/C++"); //查找"C++"在字符串中的位置
cout<<pos<<endl;
http.replace(pos, 4, ""); //从位置pos开始,之后的4个字符替换为空,即删除
cout<<http<<endl;
//找子串runoob
int first = http.find_first_of("."); //从头开始寻找字符'.'的位置
int last = http.find_last_of("."); //从尾开始寻找字符'.'的位置
cout<<http.substr(first+1, last-first-1)<<endl; //提取"runoob"子串并打印
return 0;
}
C++ 中输入的方式其实还有很多,下面来介绍一种与 C 语言中 getchar()
类似的。
cin.getline();
cin.getline()
是在输入一段字符完成后开始读取数据(注意,是输入完成后,以Enter
为结束标志)
下面是一实例:输入一串字符,编程统计其中的数字个数和英文字母个数。输入的字符以#
为结束标志。
#include<iostream>
using namespace std;
#define N 100
int main()
{
char X[N];
cin.getline(X,N); //以cin.getline形式输入
int a=0,b=0;
for(int i=0;i<N;i++)
{
if(X[i]=='#') //为#为结束标志
break;
if(X[i]>='0'&&X[i]<='9')
a++; //统计数字个数
if((X[i]>='a'&&X[i]<='z')||(X[i]>='A'&&X[i]<='Z'))
b++; //统计英文字母个数
}
cout<<a<<endl<<b<<endl;
return 0;
}
字符串与vector
字符串字面值与标准库string
不是同一种类型
string s("hello");
cout<<s.size()<<endl; //OK
cout<<"hello".size()<<endl; //ERROR
cout<<s+"world"<<endl; //OK
cout<<"hello"+"world"<<endl; //ERROR
strlen、sizeof与size()求字符串长度的区别
cout<<strlen("123")<<endl; //返回 3
cout<<sizeof("123")<<endl; //返回 4
string s = "123";
cout<<s.size()<<endl; //返回 3
标准string
库中的getline
函数返回时会丢弃换行符
const iterator与const_iterator的区别
vector<int>::const_iterator //不能改变指向的值,自身的值可以改变
const vector<int>::iterator //可以改变指向的值,自身的值不能改变
const vector<int>::const_iterator //自身的值和指向的值都是只读的
任何改变vector
长度的操作都会使已存在的迭代器失效。如:在调用push_back
之后,就不能再信赖指向vector
的迭代器了
vector<int> ivec;
ivec.push_back(10);
vector<int>::iterator it = ivec.begin();
cout<<*it<<endl;
ivec.push_back(9);
cout<<*it<<endl; //迭代器已经失效
1、sizeof
操作符的结果类型是 size_t
,它在头文件中typedef
为 unsigned int
类型。该类型保证能容纳实现所建立的最大对象的字节大小。
2、sizeof
是运算符,strlen
是函数。
3、sizeof
可以用类型做参数,strlen
只能用 char*
做参数,且必须是以\0
结尾的。sizeof 还可以用函数做参数,比如:
short f();
printf("%d\n", sizeof(f()));
输出的结果是sizeof(short)
,即 2。
4、数组做 sizeof
的参数不退化,传递给 strlen
就退化为指针了。
5、大部分编译程序在编译的时候就把 sizeof
计算过了,是类型或是变量的长度,这就是 sizeof(x)
可以用来定义数组维数的原因。
char str[20]="0123456789";
int a=strlen(str); // a=10;
int b=sizeof(str); // 而 b=20;
6、strlen
的结果要在运行的时候才能计算出来,是用来计算字符串的长度,不是类型占内存的大小。
7、sizeof
后如果是类型必须加括弧,如果是变量名可以不加括弧。这是因为 sizeof
是个操作符不是个函数。
8、当适用一个结构类型或变量时, sizeof 返回实际的大小;当适用一静态地空间数组, sizeof
归还全部数组的尺寸;sizeof
操作符不能返回动态地被分派了的数组或外部的数组的尺寸。
数组作为参数传给函数时传的是指针而不是数组,传递的是数组的首地址, 如:
fun(char [8])
fun(char [])
都等价于
fun(char *)
在 C++
里参数传递数组永远都是传递指向数组首元素的指针,编译器不知道数组的大小。
如果想在函数内知道数组的大小, 需要这样做:
进入函数后用memcpy
拷贝出来,长度由另一个形参传进去
fun(unsiged char *p1, int len)
{
unsigned char* buf = new unsigned char[len+1]
memcpy(buf, p1, len);
}
看了上面的详细解释,发现两者的使用还是有区别的,从这个例子可以看得很清楚:
char str[20]="0123456789";
int a=strlen(str); // a=10; >>>> strlen 计算字符串的长度,以结束符 0x00 为字符串结束。
int b=sizeof(str); // 而 b=20; >>>> sizeof 计算的则是分配的数组 str[20] 所占的内存空间的大小,不受里面存储的内容改变。
上面是对静态数组处理的结果,如果是对指针,结果就不一样了。
char* ss = "0123456789";
sizeof(ss)
结果 4 ===》ss 是指向字符串常量的字符指针,sizeof 获得的是一个指针的之所占的空间,应该是长整型的,所以是 4。
sizeof(*ss)
结果 1 ===》*ss 是第一个字符 其实就是获得了字符串的第一位 ‘0’ 所占的内存空间,是 char 类型的,占了 1 位
strlen(ss)
= 10 ===》 如果要获得这个字符串的长度,则一定要使用 strlen
。
strlen
用来求字符串的长度;
而 sizeof
是用来求指定变量或者变量类型等所占内存大小。
关于字符数组为什么可以以数组名来用cout输出数组内容,而普通数组不行。
先上范例:
int a[10] = {1,2,3,6,7};
char b[6] = {'h','a','p','p','y','\0'};
char c[] = "happy";
cout<<a<<endl;
cout<<b<<endl;
cout<<c<<endl;
输出结果为:
0x22fe6c
happy
happy
从以上范例可以看出,普通数组中以数组名用cout来输出,只会得到一串地址;用字符数组则会输出数组中的内容。
那为什么会这样呢?
答案:因为char
型数组中的每一个元素都是一字节,所以每一个字符之间的地址都是 +1 的是连续的,所以当 cout 输出时读到字符数组中的 \0 便停止输出; 而 int 数组每个元素占 4 个字节所以数个数组中每个元素地址的间隔是 4,但其实它也是连续的,出现乱码是因没找到结束符。
Vs2017
使用 strcpy
的时候会报错,提示 strcpy
是不安全的,需要用 strcpy_s
代替。
#include "pch.h"
#include <iostream>
#include <cstring>
using namespace std;
int main()
{
char str1[11] = "hello";
char str2[12] = "world";
char str3[11];
int len;
//复制str1到str2
strcpy_s(str3, str1);
cout << "strcpy(str3, str1):" << str3 << endl;
return 0;
}
创建详解
#include<iostream>
#include<algorithm>
#include<cmath>
#include<cstring>
#include<cstdio>
using namespace std;
int main()
{
//1. 字符串的创建
cout<<"第一:字符串的创建!\n\n";
string a(4,'a');
cout<<"1.以 a 为原字符 4单位大小\n\n";
cout<<"string a(4,'a');\ncout<<a<<endl;\n";
cout<<a<<endl<<endl;
cout<<"2.任意大小的字符串\n\n";
cout<<"string b(\"bbbbbb\");\ncout<<b<<endl;\n";
string b("bbbbbb");
cout<<b<<endl<<endl;
cout<<"3.把某一字符串的某一部分\n(0位置开始4个长度)给c\n\n";
cout<<"string c(a,0,4) ;\ncout<<c<<endl;\n";
string c(a,0,4) ;
cout<<c<<endl<<endl;
cout<<"4. 10长度原长度不足补*;\n\n";
cout<<"c.resize(10,'*');\ncout<<c<<endl;\n";
c.resize(10,'*');
cout<<c<<endl<<endl;
system("pause");
system("cls");
return 0;
}
assign()
、 copy()
详解:
#include<iostream>
#include<algorithm>
#include<cmath>
#include<cstring>
#include<cstdio>
using namespace std;
int main()
{
cout<<"第二: 字符串的赋值 assign();"<<endl;
cout<<"1.感觉就像是append不过是抹除-覆盖\n";
cout<<"string e;\nchar f[10]=\"123456\"\ne.assign(f);\ne+=' ';\ncout<<e<<endl<<endl;\n";
string e;
char f[10]="123456";
e.assign(f);
e+=' ';
cout<<e<<endl<<endl;
cout<<"2.string区间 赋值都类似吧\n";
cout<<"e.assign(f,3,3);\ne+=' ';\ncout<<e<<endl<<endl;\ne.assign(f,3);\ncout<<e<<endl;\n";
e.assign(f,3,3);
e+=' ';
cout<<e<<endl;
e.assign(f,3);
cout<<e<<endl<<endl;
cout<<"3.某字符串char型 全部\n";
cout<<"char ssr[10]=\"asdqwezxc\";\ne.assign(ssr);\ncout<<ssr<<endl;\n";
char ssr[10]="asdqwezxc";
e.assign(ssr);
cout<<ssr<<endl<<endl;
cout<<"4.某字符串char型 前num个\n";
cout<<"e.assign(ssr,4);\ncout<<e<<endl;\n";
e.assign(ssr,4);
cout<<e<<endl<<endl;
cout<<"5.某字符赋值\n";
cout<<"赋值3个6\n";
e.assign(3,'6');
cout<<e<<endl<<endl;
cout<<"copy() 将d中的2位置开始的12个字符覆盖到char型数组ss上\n 必须为-> char型 <-否则报错";
cout<<" char ss[10]=\"123\";\n string dd;\nd.copy(ss,12,2);\ncout<<ss<<endl;\n";
char ss[15]="123";
string dd("abcdefghijklmn");
dd.copy(ss,12,2);
cout<<ss<<endl<<endl;
system("pause");
system("cls");
return 0;
}
append() 详解及其扩展(int, char):
#include<iostream>
#include<algorithm>
#include<cmath>
#include<cstring>
#include<cstdio>
using namespace std;
int main()
{ cout<<"第三: 字符串的添加与复制 append();\nstring d(a);\ncout<<d<<endl;\n//或者 d=d+a;/nd.append(b);\n";
cout<<"1.在d的末尾添加字符串a\n\n";
string d(a);
d.append(b);
cout<<d<<endl<<endl;
cout<<"2.在d的末尾添加字符串/nb(0位置开始,2个长度)的数据\n\n";
cout<<"d.append(b,0,2);\ncout<<d<<endl;\n";
d.append(b,0,2);
cout<<d<<endl<<endl;
cout<<"3.添加4个 ~ 字符\n\n";
cout<<"d.append(4,'~');\n";
cout<<"cout<<d<<endl<<endl;\n";
d.append(4,'~') ;
cout<<d<<endl<<endl;
system("pause");
system("cls");
cout<<"4. int 与 char 型添加 (好高兴自己想到的int ^_^ )\n";
cout<<"char app[100]=\"aaabbb\";";
cout<<"\nstring charr(\"-_-\");\n";
cout<<"charr.append(app);\ncout<<charr<<endl\n";
char app[100]="aaabbb";
string charr("-_- ");
charr.append(app);
cout<<charr<<endl<<endl;
cout<<"charr.append(app,4);\ncout<<charr<<endl<<endl;\n";
charr.append(app,4);
cout<<charr<<endl<<endl;
cout<<"char型数组全部,char型数组的前4个\n\n****如果要添加中间***\n";
cout<<"string tmp;\nstring tmp;tmp.assign(app);\ncharr.assign(\"\");\ncharr.append(tmp,1,4);\ncout<<charr<<endl<<endl;\n";
string tmp;
tmp.assign(app);
charr.assign("");
charr.append(tmp,1,4);
cout<<charr<<endl<<endl;
cout<<"5.int double 等等 通过 sprintf() <cstdio>作为转接\n";
cout<<"int aaa=15314;\ndouble bbb=3.1415;\nchar aa[10];\nsprintf(aa,\"%d\",aaa);\ncharr.append(aa,0,4);\nsprintf(aa,\"%f\",bbb);\ncharr.append(aa,0,4);\ncout<<charr<<endl;\n";
int aaa=15314;
double bbb=3.1415;
char aa[10];
sprintf(aa,"%d",aaa);
charr.append(aa,0,4);
sprintf(aa,"%f",bbb);
charr.append(aa,0,4);
cout<<charr<<endl<<endl;
system("pause");
system("cls");
return 0;
}