假如我们有这么一段代码
#include <stdio.h> int main(void){ char c1[2]; scanf("%s",c1); char c2[20]; scanf("%s",c2); printf("%s %s\n",c1,c2); return 0; }
char类型数组在栈区开辟了自己的空间。当我们输入数据
得出的结果是我们输入的数据,但是奇怪的是我们这里c1的长度限定为2为什么还是把输入的数据全部存储并且输出呢?因为它在内存空间中是连续存储的,我们在刚才使用的scanf的第二个参数中使用的是数组c1的首地址,在接收到一组数据时,它边按照连续的内存依次存储下去,且因为是字符数组,因此在后面追加一个空字符。在printf函数中的输出格式为%s,在读取c1的首地址上面指向变量的值的时候,遇到空字符则停止读取。所以我们刚刚即使输入了长度大于2的数据,我们仍然能读出相应的数据,但这种行为是未定义的,假如在后面的内存中有其他数据就会造成一些错误。
假如我们调换c1和c2数组的大小
#include <stdio.h> int main(void){ char c1[20]; scanf("%s",c1); char c2[2]; scanf("%s",c2); printf("%s %s\n",c1,c2); return 0; }
为什么输出的结果与前面得出结论的不一样呢?同样的我们需要对存储的内存空间进行理解。我希望通过图示帮助大家理解。
c1数组的数据我们先存储在栈空间中,我们先是给c1中的字符变量赋值,后来的c2开辟的空间与c2的空间连续,再给c2赋值,值得注意的是这里因为存储的内存连续,所以在给c2赋值的时候,因为我们这里通过键盘输入的数据是大于c2数组所占空间大小的所以会把多余的数据存储在c1中,在printf语句中,printf("%s %s\n",c1,c2); c1数组里面的一些值此时已经被覆盖,所以输出c1的值就不会是我们第一步存储的数据abcd而是34。