关于char[]数组通过scanf赋值使用上的一些问题。

假如我们有这么一段代码

#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;
}

Schema字符串数组怎么 字符串数组scanf赋值_数组

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;
}

Schema字符串数组怎么 字符串数组scanf赋值_赋值_02

为什么输出的结果与前面得出结论的不一样呢?同样的我们需要对存储的内存空间进行理解。我希望通过图示帮助大家理解。

Schema字符串数组怎么 字符串数组scanf赋值_赋值_03

c1数组的数据我们先存储在栈空间中,我们先是给c1中的字符变量赋值,后来的c2开辟的空间与c2的空间连续,再给c2赋值,值得注意的是这里因为存储的内存连续,所以在给c2赋值的时候,因为我们这里通过键盘输入的数据是大于c2数组所占空间大小的所以会把多余的数据存储在c1中,在printf语句中,printf("%s %s\n",c1,c2); c1数组里面的一些值此时已经被覆盖,所以输出c1的值就不会是我们第一步存储的数据abcd而是34。