现在的位置: 首页 > 综合 > 正文

char str[]与char *str的区别

2018年05月05日 ⁄ 综合 ⁄ 共 2222字 ⁄ 字号 评论关闭

           在C语言中,对字符串的操作主要有两种方式,一是使用字符数组,char str[];二是使用字符指针。那么二者有什么区别呢?下面将分述二者的使用,最后进行比较。


一、字符数组

        使用char str[]定义一个字符数组str,中括号内可以写上数字表示数组大小,也可以不写。如果不写数字,则必须为字符数组提供初始值,以便编译器进行内存分配。

可以使用字符串字面值(string literal)来初始化字符数组,也可使用字符字面值(character literal)初始化,如:

char str1[10]="Hello";
char str2[]="World";
char str3[]={'H','e','l','l','o'};

只能对字符数组元素的赋值,而不能用赋值语句对整个数组赋值,如:

char str4[10];
str4={'H','e','l','l','o'};    // 错误
str4="Hello";                  // 错误
str4[0]='H';str4[1]='e';str4[2]='l';str4[3]='l';str4[4]='o';   // 正确

可以使用循环将字符数组中的字符一个一个输出,也可以使用cout<<str1直接输出整个数组。

需要注意的是,上述代码中str1和str2是C风格字符串,而str3不是。C风格字符串,是指以\0结尾的字符数组。C++为了兼容C,而保留了C中字符串的使用方法。

str1和str2使用字符串字面值进行初始化,字符串字面值使用\0表示字符串结束。因此str2长度为6,需要将\0计算在内。使用strlen函数,计算的是字符串的实际长度,不包含\0。

而str3则不一样,它没有\0作为结束标志,因而不是C风格字符串,使用cout<<str3可能会出现意想不到的结果。


二、字符指针

可以使用char *str指向一个字符串。如:

char *ptr="C++";
char strArr[]="C++";
char *ptr2=strArr;

使用cout<<ptr即可输出整个字符串,而使用cout<<*ptr则输出字符串的首字符。

字符指针也可指向C风格字符串,如ptr就是指向的C风格字符串。如果让ptr指向上节中的str3,输出ptr会出现同样的意想不到的结果。毕竟数组名其实就是一种指针。


三、区别

前面简单介绍了一下两种操作字符串的方法,这部分进行比较,是本文的重点。如下代码:

	char s[]="abc";
	char *ptr="abc";
	cout<<s<<endl;               // abc
	cout<<*s<<endl;              // a
	cout<<&s<<endl;              // 地址
	cout<<(s+1)<<endl;           // bc
	cout<<*(s+1)<<endl;          // b
	cout<<&s[1]<<endl<<endl;     // a
	cout<<ptr<<endl;             // abc
	cout<<*ptr<<endl;            // a
	cout<<&ptr<<endl;            // 地址
	cout<<(ptr+1)<<endl;         // bc
	cout<<*(ptr+1)<<endl;        // b
	cout<<&ptr[1]<<endl;         // a


这些代码应该能够说明char s[]和char *ptr之间的相似点了。它们都是指向字符串的指针。

下面说二者的不同之处。如下一段代码:

	char ss[]="C++";
	ss[0]='c';                  // 合法
	char *p="C++";
	p[0]='c';                   // 合法但不正确


该段代码在VS2010下编译可以通过,但是运行时程序会停止工作,为什么呢?原因在于p[0]='c'这一语句。该语句试图修改p指向的字符串的首个字符,出现了错误。

原因在于两种方式对字符数组操作的机制不同。使用char *p="C++"语句后,编译器在内存的文字常量区分配一块内存,保存”C++“这一字符串字面值,然后在栈上分配内存保存p,p的内容为"C++"的地址。p[0]='c'试图修改常量”C++“,程序当然就会崩溃了。而char ss[]="C++"语句,定义了一个数组,编译器为其在栈上分配了内存空间,因而可以进行修改操作。

因此,可以总结如下:

(1)char ss[]定义了一个数组,ss可认为是一个常指针,ss不可改变,但ss指向的内容可以发生改变。

(2)char *p定义了一个可变指针,p可以指向其它对象。但对于char *p=”abc“这样的情况,p指向的是常量,故内容不能改变。


如下代码进一步说明char ss[]和char *p的区别:

char *strA()
{
    char str[]="Hello";
    return str;
}

调用该函数,不一定能够得到正确的结果。因为str定义了一个局部数据,是局部变量,存在于函数strA中的栈帧中。当函数调用完成后,栈帧恢复到函数strA调用前的状态,临时空间被重置,为函数分配的栈空间被收回,str所指向的地址也就不存在了。

将上述代码修改:

char *strA()
{
    char *str="Hello";
    return str;
}

该函数能够正常运行,因为str指向的字符串字面值被保存在只读的数据段,是全局的,当函数调用完成后,str指向的地址未发生变化。

综上,可以看出使用char []较容易出错,可能出现不确定的结果。C++提供的string类相比之下,要安全的多了。

参考自:http://blog.csdn.net/yahohi/article/details/7427724

——The End——

抱歉!评论已关闭.