1、字符数组的定义与初始化
字符数组的初始化,最容易理解的方式就是逐个字符赋给数组中各元素。
char str[10]={ 'I',' ','a','m',' ',‘h’,'a','p','p','y'};
即把10个字符分别赋给str[0]到str[9]10个元素
如果花括号中提供的字符个数大于数组长度,则按语法错误处理;若小于数组长度,则只将这些字符数组中前面那些元素,其余的元素自动定为空字符(即 '/0' )。
2、字符数组与字符串
在c语言中,将字符串作为字符数组来处理。(c++中不是)
在实际应用中人们关心的是有效字符串的长度而不是字符数组的长度,例如,定义一个字符数组长度为100,而实际有效字符只有40个,为了测定字符串的实际长度,C语言规定了一个“字符串结束标志”,以字符'/0’代表。如果有一个字符串,其中第10个字符为'/0',则此字符串的有效字符为9个。也就是说,在遇到第一个字符'/0'时,表示字符串结束,由它前面的字符组成字符串。
系统对字符串常量也自动加一个'/0'作为结束符。例如"C Program”共有9个字符,但在内存中占10个字节,最后一个字节'/0'是系统自动加上的。(通过sizeof()函数可验证)
有了结束标志'/0'后,字符数组的长度就显得不那么重要了,在程序中往往依靠检测'/0'的位置来判定字符串是否结束,而不是根据数组的长度来决定字符串长度。当然,在定义字符数组时应估计实际字符串长度,保证数组长度始终大于字符串实际长度。(在实际字符串定义中,常常并不指定数组长度,如char str[ ])
说明:'/n’代表ASCII码为0的字符,从ASCII码表中可以查到ASCII码为0的字符不是一个可以显示的字符,而是一个“空操作符”,即它什么也不干。用它来作为字符串结束标志不会产生附加的操作或增加有效字符,只起一个供辨别的标志。
对C语言处理字符串的方法由以上的了解后,再对字符数组初始化的方法补充一种方法——即可以用字符串常量来初始化字符数组:
char str[ ]={"I am happy"}; 可以省略花括号,如下所示
char str[ ]="I am happy";
注意:上述这种字符数组的整体赋值只能在字符数组初始化时使用,不能用于字符数组的赋值,字符数组的赋值只能对其元素一一赋值,下面的赋值方法是错误的
char str[ ];
str="I am happy"; //整体赋值
不是用单个字符作为初值,而是用一个字符串(注意:字符串的两端是用双引号“”而不是单引号‘’括起来的)作为初值。显然,这种方法更直观方便。(注意:数组str的长度不是10,而是11,这点请务必记住,因为字符串常量"I am happy"的最后由系统自动加上一个'/0')
因此,上面的初始化与下面的初始化等价
char str[ ]={'I',' ','a','m',' ','h','a','p','p','y','/0'};
而不与下面的等价
char str[ ]={'I',' ','a','m',' ','h','a','p','p','y'};
前者的长度是11,后者的长度是10.
说明:字符数组并不要求它的最后一个字符为'/0',甚至可以不包含'/0',向下面这样写是完全合法的。
char str[5]={'C','h','i','n','a'};
++++++++
可见,用两种不同方法初始化字符数组后得到的数组长度是不同的。
#include <stdio.h>
void main(void)
{
char c1[]={'I',' ','a','m',' ','h','a','p','p','y'};
char c2[]="I am happy";
int i1=sizeof(c1);
int i2=sizeof(c2);
printf("%d/n",i1);
printf("%d/n",i2);
}
结果:10 11
3、字符串的表示形式
在C语言中,可以用两种方法表示和存放字符串:
(1)用字符数组存放一个字符串
char str[ ]="I love China";
(2)用字符指针指向一个字符串
char* str="I love China";
对于第二种表示方法,有人认为str是一个字符串变量,以为定义时把字符串常量"I love China"直接赋给该字符串变量,这是不对的。
C语言对字符串常量是按字符数组处理的,在内存中开辟了一个字符数组(字符串在栈中,可读写)用来存放字符串常量,程序在定义字符串指针变量str时只是把字符串首地址(即存放字符串的字符数组的首地址)赋给str(此时,字符串值在 内存的 只读区 ,linux下为 .rodata 区,为只读)。
两种表示方式的字符串输出都用
printf("%s/n",str);
%s表示输出一个字符串,给出字符指针变量名str(对于第一种表示方法,字符数组名即是字符数组的首地址,与第二种中的指针意义是一致的),则系统先输出它所指向的一个字符数据,然后自动使str自动加1,使之指向下一个字符...,如此,直到遇到字符串结束标识符 " /0 "。
4、对使用字符指针变量和字符数组两种方法表示字符串的讨论
虽然用字符数组和字符指针变量都能实现字符串的存储和运算,但它们二者之间是有区别的,不应混为一谈。
4.1、字符数组由若干个元素组成,每个元素放一个字符;而字符指针变量中存放的是地址(字符串/字符数组的首地址),绝不是将字符串放到字符指针变量中(是字符串首地址)
4.2、赋值方式:
对字符数组只能对各个元素赋值(一个一个的赋值),不能用以下方法对字符数组赋值
char str[14];
str="I love China"; (但在字符数组初始化时可以,即char str[14]="I love China";)
而对字符指针变量,采用下面方法赋值:
char* a;
a="I love China";
或者是 char* a="I love China"; 都可以
4.3、(初始化)
对字符指针变量赋初值:
char* a="I love China";
等价于:
char* a;
a="I love China";
而对于字符数组的初始化
char str[14]="I love China";
不能等价于:
char str[14];
str="I love China"; (这种不是初始化,而是赋值,而对数组这样赋值是不对的)
4.4、如果定义了一个字符数组,那么它有确定的内存地址;而定义一个字符指针变量时,它并未指向某个确定的字符数据,并且可以多次赋值。
5、字符串处理函数
5.1
char *strcat(char *str1,const char *2 );
char *strcat(char *strDestination,const char *strSource );
功能:函数将字符串str2 连接到str1的末端,并返回指针str1
注:连接前两个字符串的后面都有一个' /0 ',连接时将字符串1后面的 ' /0 ‘去掉,只在新串最后保留一个 ' /0 ‘
5.2
char *strcpy(char *str1,const char *2 );
char *strcpy(char *strDestination,const char *strSource );
功能:复制字符串strSource中的字符到字符串strDestination,包括空值结束符。返回值为指针strDestination。
注:1、“字符数组1”必须写成数组名形式,“字符串2"可以是字符数组名,也可以是一个字符串常量
2、复制时连同字符串后面的 ' /0 ' 一起复制到数组1中
3、不能用赋值语句直接将一个字符串常量或者字符数组直接赋给一个字符数组(同普通变量数组是一样的),而只能用strcpy函数处理。
4、可以用strcpy函数将字符串2中的前若干个字符复制到字符数组1中去。
关于字符数组和字符串指针的讨论问题:
本帖来源:http://www.9php.com/FAQ/cxsjl/c/2008/12/2534264136846.html
今天在编程时发现这样一个现象
char str1[]="abcd";
char str2[]="efgh";
*str1=*str2;
这个操作是合法的,结果也是正确的;
但是如果用字符串指针声明:
char *str1="abcd";
char *str2="efgh";
*str1=*str2;
运行时会出现内存错误
编译的平台:gcc 3.2.2 以及dev-c++都试验过
比较疑惑,感谢大家的指教^^
2008-12-13 19:10Fixend简单一点说, “abc”是const char *型的,数据放在只读的内存区。
而char[]在栈,可写。
2008-12-13 19:14aobai这个在《c专家编程》上面好像是说的未定义呀~
2008-12-13 20:18Godbach
今天在编程时发现这样一个现象 char str1[]="abcd"; char str2[]="efgh"; *str1=*str2; 这个操作是合法的,结果也是正确的;[/quote] 这两个字符串都是保存在栈上,可以修改其中的内容。其中,*str1=*str2相当于把str2字符串的第一个字符'e'赋给了str1的第一个字符,因此,str1就变成了"ebcd"
我为刚才的发帖感到脸红~~,你是对的, //字符数组 char str1[]="abcd"; char str2[]="efgh"; *str1=*str2; 的汇编代码是: .file "2x.c" .section .rodata .LC0: .string "abcd" .LC1: .string "efgh" .text .globl main .type main, @function main: leal 4(%esp), %ecx andl $-16, %esp pushl -4(%ecx) pushl %ebp movl %esp, %ebp pushl %ecx subl $16, %esp #put .LC0 in -9(%ebp) to-5(%ebp) movl .LC0, %eax movl %eax,-9(%ebp) movzbl .LC0+4, %eax movb %al,-5(%ebp) #put .LC1 in -14(%ebp) to-10(%ebp) movl .LC1, %eax movl %eax,-14(%ebp) movzbl .LC1+4, %eax movb %al,-10(%ebp) #*str1 =*str2 #not arrange address ,but arrange the first char; movzbl -14(%ebp), %eax movb %al,-9(%ebp) # all done addl $16, %esp popl %ecx popl %ebp leal -4(%ecx), %esp ret .size main,.-main .ident "GCC: (GNU) 4.2.3 (Debian 4.2.3-5)" .section .note.GNU-stack,"",@progbits //字符串指针 char *str1="abcd"; char *str2="efgh"; *str1=*str2; 的汇编代码如下: .file "3x.c" .section .rodata .LC0: .string "abcd" .LC1: .string "efgh" .text .globl main .type main, @function main: leal 4(%esp), %ecx andl $-16, %esp pushl -4(%ecx) pushl %ebp movl %esp, %ebp pushl %ecx subl $16, %esp movl $.LC0,-12(%ebp) movl $.LC1,-8(%ebp) movl -8(%ebp), %eax movzbl (%eax), %edx movl -12(%ebp), %eax movb %dl,(%eax) addl $16, %esp popl %ecx popl %ebp leal -4(%ecx), %esp ret .size main,.-main .ident "GCC: (GNU) 4.2.3 (Debian 4.2.3-5)" .section .note.GNU-stack,"",@progbits 声明为指针的时候,在栈上保存的是str的存储地址,而str是声明在.rodata区的, 所以最后 movl -12(%ebp), %eax movb %dl, (%eax) 相当于尝试把str2的首字符写入 .rodata区,所以就SIGSEGV了, 而声明为char []的时候,其实就是把字符在栈上操作,因此能正常执行(读写), 虽然这种操作没啥意义~~ 2008-12-13 22:20liu2g[quote]原帖由 Godbach 于 2008-12-13 20:18 发表 对于字符数组 这两个字符串都是保存在栈上,可以修改其中的内容。其中,*str1=*str2相当于把str2字符串的第一个字符'e'赋给了str1的第一个字符,因此,str1就变成了"ebcd" 以上两个字符串都是存在于静态内存区域。对于该 ... |
赞成!!