gets获取一行数据包括末尾的回车都获取。
所以输入一行数据,然后回车结束,系统就默认这一行数据都是输入,除开末尾的回车字符。
#include<stdio.h>
#include<string.h>
main()
{
char a1[5],a2[5];
gets(a1);
gets(a2);
puts(a1);
puts(a2);
printf("%d\n",strlen(a2));
}
当我输入:
pen book
penbook
时
为什么输出是:
pen bopenbook
penbook
a1的数组不是定义5个字符吗?
为什么输出是这么多?
使用 gets 必须保证输入长度不会超过缓冲区,否则会出错,但因编译器内存对齐的原因,有时缓冲区的溢出并不表现出来从而导致奇怪的结果(逻辑上溢出,物理上并未溢出)
char a1[5],a2[5]; // 一般来说,编译器会使a1,a2分配连续的空间,但因为内存对齐的原因,在TC下字长为16位,实际上每个数组占6个字节
当输入pen book 时,a1 已经溢出,并占了a2的空间(a2未溢出),由于 a2 是我们自己的空间,所以不会出现“非法指令,内存不能为writte”之类的错误提示, 此时 a1, a2 内容如下
a1 : 'p', 'e', 'n', ' ', 'b', 'o'
a2 : 'o', 'k', '\0' ,... // 后面随机
接着输入 penbook, 此时会从 a2
开始
存储, a1, a2 内容如下
a1 : 'p', 'e', 'n', ' ', 'b', 'o'
a2 : 'p', 'e', 'n', 'b', 'o', 'o', 'k', '\0'
// 此时a2在逻辑和物理上都溢出了,是否会出理错误就看运行了, 要看a2后的内存使用情况
所以输出结果如下
pen bopenbook
penbook
在VC6下,内存对齐的结果是 8 个字节,分析同上
输入
pen book
penbook
则
a1 : 'p', 'e', 'n', ' ', 'b', 'o', 'o', 'k', '\0' // 溢出
a2 : 'p', 'e', 'n', 'b', 'o', 'o', 'k', '\0' // 逻辑上溢出,物理上没有
接前面的分析,输出结果应该是
pen bookpenbook
penbook
而事实下输出的是
pen book
penbook
原因是VC6下内存分配顺序不同, 此时a2在前,a1在后,所以a1的输出是随机的,可能出错,也可能正常输出,要看a1后面内存的使用情况
如果将 char a1[5],a2[5]; 改成 char a2[5], a1[5]; 则输出就用TC下的情况一样了:
pen bookpenbook
penbook