小朋友学C语言(42):gets和fgets

一、gets()函数

原型:char *gets(char *str); 头文件:stdio.h

例1

#include <stdio.h>

int main()
{
    char str[10];
    gets(str);
    puts(str);

    return 0;
}

(1)在Windows系统中的运行结果

hello
hello

(2)在Linux中用GCC进行编译

noilinux@ubuntu:~/Desktop$ gcc test.c -o test
test.c: In function ‘main’:
test.c:6:5: warning: ‘gets’ is deprecated (declared at /usr/include/stdio.h:638) [-Wdeprecated-declarations]
     gets(str);
     ^
/tmp/cc0hPgqA.o:在函数‘main’中:
test.c:(.text+0x1d): 警告: the `gets' function is dangerous and should not be used.

运行结果

noilinux@ubuntu:~/Desktop$ ./test
hello
hello

例2

#include <stdio.h>

int main(void)
{
    char c;
    char s[3];
    scanf("%c",&c);
    getchar();      // 过滤回车
    gets(s);

    printf("string=%s\nchar=%c\n", s, c);

    return 0;
}

运行结果 在windows下输入:

a
hi

输出

string=hi
char=a

在windows下重新输入

a
uvwxyz

输出

string=uvw
char=x

这里可以看出来,定义了s的长度为3,但是用gets()输入字符串的时候,并不会去检查字符串的长度,所以导致char的值不是a,而是”uvwxyz”中的第四个字符’x’。 不正确使用gets()函数的时候造成的危害是很大的,就像我们刚才看到的那样,a的值被字符串s溢出的值给替换了。 因为gets有不限制输入字符个数的限制,可能会导致不法分子利用这一漏洞造成缓冲区溢出,从而达到破坏的目的。《C Primer Plus》中提到蠕虫病毒就是利用这一漏洞来攻击操作系统。 出于安全考虑,用fgets()来代替gets()。

二、fgets()函数

原型:char * fgets(char * s, int n,FILE *stream); 头文件:stdio.h fgets()函数读取到它所遇到的第一个换行符的后面,或者读取比字符串的最大长度少一个的字符,或者读取到文件结尾。然后fgets()函数向末尾添加一个空字符以构成一个字符串。如果在达到字符最大数目之前读完一行,它将在字符串的空字符之前添加一个换行符以标识一行结束。

例3

#include <stdio.h>
#define len 5

int main()
{
    char c;
    char s[len];
    scanf("%c", &c);
    getchar();  // 过滤回车
    fgets(s, len, stdin);

    printf("string=%s\nchar=%c\n", s, c);

    return 0;
}

运行结果 输入:

a
uvwxyz

输出:

string=uvwx
char=a
这里string=uvwx的后面其实还有一个空字符’\0’没有显示出来。

例4

#include <stdio.h>
#include <string.h>
#define len 100

int main()
{
    // stdin,标准输入流,默认是键盘,重定向到文件title.in,scanf或fgets从title.in中读取数据
    freopen("title.in", "r", stdin);
    // stdout,标准输出流,默认是显示器,重定向到title.out,printf把数据写到title.out中
    freopen("title.out", "w", stdout);

    char s[len];
    fgets(s, len, stdin);
    int total = 0;

    // 这里只能用strlen,不能用sizeof
    // 因为sizeof(s)=sizeof(s)/sizeof(char)=100
    for(int i = 0; i < strlen(s); i++)
    {
        if(s[i] != ' ')
        {
            total++;
        }
    }

    printf("%d", total);
    return 0;
}

原文发布于微信公众号 - KidsCode少儿编程(gh_de7b45c40e8b)

原文发表时间:2018-11-23

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏coder修行路

Nginx location 匹配顺序整理

Nginx location模块整理 具体的Nginx安装就不在这里描述了,这里只是为了对location的描述 Nginx环境 a. 查看当前系统cat /e...

3377
来自专栏机器学习从入门到成神

队列和栈面试题(一)— 请编写一个程序,按升序对栈进行排序,要求最多只能使用一个额外的栈存放临时数据

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/sinat_35512245/articl...

1212
来自专栏python3

python3--模块collections,time,random,sys

有如下值集合[11,22,33,44,55,66,77,88,99,90......],将所有大于66的值保存至字典的第一个key中,小于66的值保存至第二个k...

1002
来自专栏李航的专栏

Shell 主要逻辑源码级分析:SHELL 运行流程 (1)

分享一下在学校的时候分析shell源码的一些收获,帮助大家了解shell的一个工作流程,从软件设计的角度,看看shell这样一个历史悠久的软件的一些设计优点和缺...

2.1K0
来自专栏海说

早期(编译期)优化

  相当多新生的java语法特性,都是靠编译器的“语法糖”来实现,而不是依赖虚拟机的底层改进来支持,java中即时编译器地运行期的优化过程对于程序运行来说更重要...

2040
来自专栏软件开发 -- 分享 互助 成长

模板方法模式

一、简介 1、模板方法模式定义一个操作中的算法骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。 2、说白...

1937
来自专栏开源优测

JMeter处理器09

前言 在jmeter中提供了两种处理器,用于修改请求数据或处理响应数据。 ? 前置处理器 后置处理器 前置处理器 前置处理器是在请求发送前做相关处理。可以用于在...

3114
来自专栏SpringBoot

一个密码复杂度的验证js

在项目开发中,要求密码进行复杂度限制,现帖出来跟大家做个分享。 密码复杂要求:1、长度大于8          2、密码必须是字母大写,字母小写,数字,特殊...

4653
来自专栏决胜机器学习

《Redis设计与实现》读书笔记(二) ——Redis中的字典(Hash)

《Redis设计与实现》读书笔记(二) ——Redis中的字典(Hash) (原创内容,转载请注明来源,谢谢) 一、概述 字典,又称符号表、关联数组、映射,是一...

39810
来自专栏BinarySec

unexploitable Writeup[pwnable.tw]

最近在刷pwnable.tw(类似pwnable.kr,不过是台湾的)的题,看到了一个unexploitable的题目。根据题目描述:

2652

扫码关注云+社区

领取腾讯云代金券