char*p,表示p是指向字符串的指针
char arr[],表示arr是一个字符串数组
string s,表示s是一个string类的对象,有自己的成员变量和成员函数,和前两者的区别较大,在下一篇博文中详细讨论
通过下面的例子来看char*与char[]的区别:
例子1:
对比下面两个函数:
char* get_str1()
{
char str[] = {"abcd"};
return str;
}
char* get_str2()
{
char *str = {"abcd"};
return str;
}
可以发现,函数get_str1编译时有警告信息:(warning C4172: 返回局部变量或临时变量的地址),这种写法是不对的,尽管编译能通过,可以运行,但是运行的结果不对,乱码了;但是函数get_str2,即char*,就正确了。
这是因为:
在函数get_str1中,char str[] = { "abcd" },定义的是一个局部变量,该函数返回的是内部局部字符数组str的地址,当函数调用之后该数组被销毁,所以返回的指针是一个已经释放了空间的指针,指向的值不定。
在函数get_str2中,char* str = { "abcd" },先定义一个字符串常量“abcd“,再将指针str指向它,返回值是这个指针,由于字符串常量在编译时分配内存,只有程序退出时才被销毁,所以返回它的地址没有问题。
例子2:
1)char s[5];
s = "abcd"; //错误
2)char *s;
s = "abcd"; //正确
1编译不通过但是2编译通过。这是因为在1中,s是数组名,当成是常量指针,不可以作为左值修改;而2中s是一个指针变量,可以将其指向一个字符串常量。
总的来说
1)
char str[] = { "abcd" },str在栈上申请空间,将常量内容复制进来
char *str = {"abcd"},str指向该常量的地址
2)
char str[] = { "abcd" },sizeof(str) = 5,strlen(str) = 4
char *str = {"abcd"},sizeof(str) = 4(x86)或者8(x64),strlen = 4
注:指针的sizeof运算结果都是4