fgets与fget函数的区别,并使用gdb进行验证

南昌大学

废话不多说,直接开始

步骤

1.编写测试样例
2.编写Makefile进行编译
3.使用gdb工具进行调试验证

创建 my.h,此为头文件,包含函数的定义以及必要的头文件声明。

//my.h
#include <stdio.h>
#include <string.h>
#include <stdlib.h>

#include "err_exit.h"

#ifndef Y_OR_QUES
#define Y_OR_QUES
int y_or_n_(const char *);
#endif

创建p2-1.c、p2-2.c,此为测试项目

//p2-1.c
#include "my.h"

int y_or_n_ques(const char *question)
{
   fputs (question, stdout);     /* 输出提问 */
   while(1){
      int c, answer;
      fputc (' ', stdout);       /* 写一空格分开问题与回答 */
       /* 读此行第一个字符.它应当是回答字符,但也可能不是. */
      c = tolower(fgetc (stdin));
      answer = c;
      while(c != '\n' && c != EOF)   /* 忽略此行的其余字符. */
          c = fgetc(stdin);
       /* 如果是回答字符,响应回答. */
      if (answer == 'y')
          return 1;
      if (answer == 'n')
          return 0;
       /* 非回答字符,继续要求合法回答. */
      fputs ("Please answer y or n:", stdout);
   }
}
//p2-2.c
#include "my.h"
#define BUF_SIZE 8

int main(void)
{
   FILE *fd;
   int fgets_yes;
   struct iobuf {
       char buf[BUF_SIZE];
       char others[BUF_SIZE];
   }buffer;
  
   memset(&buffer,'\0',sizeof(struct iobuf)); 
   do {
       /* 请用户选择使用fgets()还是gets() */
       fgets_yes = y_or_n_ques("Should we read by fgets()?");
       fprintf(stdout,"please enter a line\n");
       if(fgets_yes) {   /* 用fgets读输入数据 */
          fgets(buffer.buf, BUF_SIZE, stdin);
          fprintf(stdout,"fgets() get string \"%s\"\n",buffer.buf);
          while(buffer.buf[strlen(buffer.buf)-1] != '\n'){ /* 一行未读完,继续读 */
             fgets(buffer.buf, BUF_SIZE , stdin); 
             fprintf(stdout,"fgets() get string \"%s\"\n",buffer.buf);
          }
       } else {   /* 用gets读输入数据 */
          gets(buffer.buf);
          fprintf(stdout,"gets() get string \"%s\"\n",buffer.buf);
       }
       /* 查看溢出情况 */
       fprintf(stdout,"buffer.others is \"%s\"\n",buffer.others);
   } while (y_or_n_ques("continue?"));
   exit(0);
}

y_or_n_ques函数的代码

//y_or_n_ques.c
#include <stdio.h>
int y_or_n_ques(const char *question)
{
    fputs (question, stdout);
    while (1){
        int c, answer;
         /* 写一空格分开问题与回答 */
        fputc (' ', stdout);
         /* 读此行中的第一个字符.它应当是回答字符,但也可能不是. */
        c = tolower (fgetc (stdin));
        answer = c;
         /* 忽略此行的其余字符. */
        while (c != '\n' && c != EOF)
            c = fgetc (stdin);
         /* 如果是回答字符,响应回答.*/
        if (answer == 'y')
            return 1;
        if (answer == 'n')
            return 0;
        /* 非回答字符,请求合法回答. */
        fputs ("Please answer y or n:", stdout);
    }
}

创建makefile文件,进行项目的编译与链接

#makefile
CC=gcc
objects=p2-1.o p2-2.o
test1:$(objects)
    $(CC) $(objects) -o test1
$%:$% my.h
    gcc -c $< -o $*.o
$%:$% my.h
    gcc -c $< -o $*.o
clean:
    rm -f *o

调试之前首先用gcc -g命令生成调试信息,否则调试失败

数据的分析与处理

1.两函数的区别
1.png
  • 使用fgets()函数:
    输入:1234567890 输出:1234567 890\n
  • 分析:p2-2.c中的memset()函数,将buffer结构体,16个地址空间用’\0’填充,fgets()函数以\0为结束符,当如上输入时,buffer.buf的前7个内存地址存入’1234567’,最后一个内存地址用’\0’填充,剩下的’890\n’留在缓冲区,留待下一次读取。第二次读取’890\n’,以本来填充的’\0’作为读取的结束符。故第二次输出,输出了换行。
    此外,fgets()函数的读取,并没有发生溢出。每次读取到n位空间时,从缓冲区读取n-1位,最后一位函数自动设为’\0’作为结束符,剩下的留待下次读取,不会出现溢出。


    2.png
  • 使用fgets()函数:
    输入:123456 输出:123456\n
  • 分析:此种情景,同上示例第二次读取


    3.png
  • 使用gets()函数:
    输入:123456789123456789 输出:123456789123456789
    溢出:9123456789
  • 分析:gets()函数以’\n’作为结束符。当如上输入时,gets()函数从首位读取,直到碰到第一个’\0’位置,并输出’\0’之前的全部字符。不会因为buffer.buf的定义大小而停止读取,此时发生溢出。(此种方式容易发生溢出攻击)
    Buffer.buf发生溢出,前八个字符被读取到buffer.buf内,剩下的依旧被gets()函数一同读取,顺位被读取到内存地址中。故buffer.others读取剩下的全部字符,包括最后的’\n’字符,并作为输出buffer.others的结束符。
2.gdb调试跟踪:
  • fgets()函数调试


    4.png

    未输入数据时,buffer.buf的内容有’\0’填充


    5.png

    输入1234567890后,读取1234567到buffer.buf前七个字节,最后一个字节被’\0’填充
    并且没有溢出。剩余数据,留待下次读取
    6.png

    下次读取890’\n’ 共四个字符,最后由’\0’作为结束符。

  • gets()函数调试
    输入1234567890后,函数读取的数据如下


    7.png

    发生溢出,数据超出BUF_SIZE的部分被读取到buffer.others中。发生溢出
    不会因为’\n’作为结束符,将其作为‘普通’字符,进行输出

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容