gets()使用问题

gets获取一行数据包括末尾的回车都获取。 所以输入一行数据,然后回车结束,系统就默认这一行数据都是输入,除开末尾的回车字符。 #include #include 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
### 关于 `gets` 函数的使用 在C语言中,`gets` 函数用于从标准输入(通常是键盘)读取一行字符直到遇到换行符或文件结束标志,并将这些字符存入指定的字符串数组中[^2]。 #### 基本语法 ```c char *gets(char *str); ``` 此函数会返回指向传入缓冲区的指针;如果发生错误或者到达EOF,则返回NULL。需要注意的是,在实际开发过程中不推荐使用该函数,因为其无法控制最大读取长度从而容易引发缓冲区溢出的安全隐患[^3]。 尽管如此,为了满足学习目的,这里给出一段简单的例子来展示如何利用 `gets` 获取用户输入: ```c #include <stdio.h> int main(void){ char inputString[100]; printf("请输入一些文字:\n"); gets(inputString); // 不建议使用的危险做法 printf("你刚刚输入的内容是:%s\n",inputString); return 0; } ``` 然而,更安全的做法是采用替代方案如 `fgets()` 来代替 `gets()` 进行字符串输入处理,这样可以有效防止潜在的风险并提高代码质量。 #### 安全替换方案——`fgets` 对于上述提到的安全性问题,可以通过调用 `fgets()` 实现相同功能的同时确保不会超出目标数组界限: ```c #include <stdio.h> #define BUFFER_SIZE 100 int main(){ char userInput[BUFFER_SIZE]; printf("请输入您的姓名:"); if (fgets(userInput, sizeof(userInput), stdin)) { // 移除可能存在的结尾换行符 size_t length = strlen(userInput); if (length > 0 && userInput[length - 1] == '\n') { userInput[--length] = '\0'; } printf("您好,%s!\n", userInput); } return 0; } ``` 通过这种方式不仅可以避免因不当使用 `gets` 而带来的风险,还能更好地管理内存边界条件下的异常情况。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值