我的程序是这样的。
int main(int argc, char **argv)
{
char str1[5],str2[4];
scanf("%s%s",str1,str2);
printf("%s\n",str1);
printf("%s\n",str2);
return 0;
},如果str1对应的输入字符串是5个的话就会出错,比如我输入abcde回车abcd回车,打印出来的是abcdeabcd,abcd.。如果我第一个字符串输入的字符个数少于5个的话就不会有问题,请问这是为什么?
你申请数组空间后,内存中存放字符的顺序是:
str1[0],str1[1],str1[2],str1[3],str1[4],str2[0],str2[1],str2[2],str2[3]
当输入字符串是:abc 123时,存储的数据str1[0]~str1[4],str2[0]~str2[3]为:
‘a','b','c',0,xx,'1','2','3',0
其中,xx表示一个不关心的任意值,也就是说输入字符串"abc"需要占用4个字节的存储空间
当输入字符串1为"abcde"时,存储的数据为:'a','b','c','d','e',0,xx,xx,xx
字符串末尾的0已经超出了str1开设的存储空间而存放到了str2[0]位置
接着再输入字符串"1234"时,str2[0]的str1字串的结尾符被字符'1'覆盖,
存储的数据为:'a','b','c','d','e',’1‘,’2‘,’3‘,’4‘,0
而且由于str2输入字符数为4个其占用空间也超出了str2的空间范围,那么上边存储的0值也不知道覆盖了什么变量的位置,导致其他变量的值被冲掉了。
当执行printf("%s\n",str1);时,printf从str1[0]开始输出的字符串(遇0结束输出)为abcde1234
当执行printf("%s\n",str2);时,printf从str2[0]开始输出的字符串(遇0结束输出)为1234
所以你输入的第1个字符串字符数少于5时,存储不会溢出,多于4个时,末尾的0就会被输入的str2所覆盖而失效,且多于5的字符也会被覆盖。scanf并不能保证输入字符的长短不会溢出,你应当在设计程序时考虑到这点,预先定义足够的字符串保存空间或用其他方法来避免这个问题的出现。
为什么str1过界的字符就一定要到str2里面呢,我看了这2个数组的地址,是连在一起的,是不是栈在分配数据的时候就默认字符数组分一起,然后整数数组分一起?
追答可以说由于你开设两个空间的定义语句的编译结果导致的,编译程序自动把你这两个空间紧接着开辟了,这是与你的变量定义语句以及编译程序相关的。但你在程序中不能假设这个是一定的,只能说碰巧了。
那是不是说每种类型的数据在栈中分配的时候都是分在连续的地址中的?
追答应该是这样的,不过也要看编辑器,还有对齐方式等因素
其实像这种问题,一般来说没必要搞的太清晰,知道个大概即可