掘根宝典之C语言字符串输入函数(gets(),fgets(),get_s())

字符串输入前的注意事项

如果想把一个字符串读入程序,首先必须预留该字符串的空间,然后用输入函数获取该字符串

 这意味着必须要为字符串分配足够的空间

不要指望计算机在读取字符串时顺便计算它的长度,然后再分配空间(计算机不会这样做,除非你编写一个处理这些任务的函数)。

假设编写了如下代码:

char *name;
scanf("%s",name);

虽然可能会通过编译(编译器很可能给出警告),但是在读入name时,name可能会擦写掉程序中的数据或代码,从而导致程序异常中止。

因为scanf()要把信息拷贝至参数指定的地址上,而此时该参数是个未初始化的指针,name可能会指向任何地方。大多数程序员都认为出现这种情况很搞笑,但仅限于评价别人的程序时。
 

最简单的方法是,在声明时显式指明数组的大小:
 

char name[81];
scanf("%s",name);


现在name是一个已分配块(81字节)的地址。还有一种方法是使用C库函数来分配内存

为字符串分配内存后,便可读入字符串。C语言提供了许多读取字符串的函数:gets(),fgets(),gets_s()函数

gets()

C语言中的gets()函数是一个标准库函数,用于从标准输入(键盘)读取一行字符并存储到指定的字符数组中。它的函数原型如下:

char *gets(char *str);

gets()函数的参数是一个字符数组,用于存储读取到的字符,返回值是读取到的字符数组的首地址。

注意:gets()函数存在一些安全性问题,因为它无法保证读取的字符数不超过指定的字符数组大小,可能导致缓冲区溢出。因此,在实际的程序开发中,最好使用更安全的替代函数fgets()来代替gets()函数。

特点

在读取字符串时,scanf()和转换说明%s只能读取一个单词。可是在程序中经常要读取一整行输入,而不仅仅是一个单词。许多年前,gets()函数就用于处理这种情况。

gets()函数简单易用,它读取整行的输入,直至遇到换行符,然后丢弃换行符,储存其余字符,并在这些字符的末尾添加一个空字符使其成为一个C字符串。

使用示例:

#include <stdio.h>
int main()
{
char words[81];gets(words) ; // 典型用法printf("s\n", words);
}

结果

//输入abcd
abcd

但是我们拿着上面这段代码去运行的时候,就会发现编译器报错或者发出警告,这是为什么呢?

缺点

问题就出现在gets唯一的参数是words,它无法检查数组是否装得下输入行。

因此gets()函数只知道数组的开始处(通过传入的数组名),但是并不知道数组中有多少个元素。

如果输入的字符串过长,会导致缓冲区溢出(buffer overflow),即多余的字符超出了指定的目标空间。

如果这些多余的字符只是占用了尚未使用的内存,就不会立即出现问题;

如果它们擦写掉程序中的其他数据,会导致程序异常中止:或者还有其他情况。

为了让输入的字符串容易溢出,把程序中的STLEN设置为5,程序的输出如下:

//输入abcd
abcd
Segmentation fault:11


“Segmentation fault”(分段错误)似乎不是个好提示,的确如此。在UNIX系统中,这条消息说明该程序试图访问未分配的内存。

C 提供解决某些编程问题的方法可能会导致陷入另一个尴尬棘手的困境。

但是,为什么要特别提到gets()函数?

因为该函数的不安全行为造成了安全隐患。

过去,有些人通过系统编程,利用gets()插入和运行一些破坏系统安全的代码。
不久,C编程社区的许多人都建议在编程时摒弃gets()。制定C99标准的委员会把这些建议放入了标准,承认了gets()的问题并建议不要再使用它。尽管如此,在标准中保留gets()也合情合理,因为现有程序中含有大量使用该函数的代码。而且,只要使用得当,它的确是一个很方便的函数。
好景不长,C11标准委员会采取了更强硬的态度,直接从标准中废除了gets()函数。然而在实际应用中,编译器为了能兼容以前的代码,大部分都继续支持gets()函数。不过,VS2022就不支持了

fgets()

过去通常用fgets()来代替gets(),fgets()函数稍微复杂些,在处理输入方面与gets()略有不同。

原型

在C语言中,fgets()函数用于从指定的输入流中读取一行字符串。它接受三个参数:输入缓冲区指针,缓冲区大小和要读取的输入流。

使用fgets()函数的语法如下:

char *fgets(char *str, int size, FILE *stream);

fgets()函数的第2个参数指明了读入字符的最大数量。

如果该参数的值是n,那么fgets()函数从输入流中读取至多n - 1个字符(因为会自动加\0),或者遇到换行符('\n')为止。

fgets()函数的第3个参数指明要读入的文件。如果读入从键盘输入的数据,则以stdin(标准输入)作为参数,该标识符定义在stdio.h中。

fgets()函数返回指向char的指针。如果一切进行顺利,该函数返回的地址与传入的第1个参数相同但是,如果函数读到文件结尾,它将返回一个特殊的指针:空指针(null pointer)。该指针保证不会指向有效的数据,所以可用于标识这种特殊情况。在代码中,可以用数字0来代替,不过在C语言中用宏NULL来代替更常见(如果在读入数据时出现某些错误,该函数也返回NULL)。

读取规则

它将读取到的字符逐个存储在字符数组中,直到达到指定的大小或者遇到换行符为止。

如果没有遇到换行符,或者输入流中没有更多字符可读,fgets()函数会在最后一个字符后面添加一个空字符('\0')(如果数组没存满,系统会自动添加\0直到装满),表示字符串的结束。

看个例子

#include<stdio.h>
int main()
{char a[10];fgets(a, 10, stdin);printf("%s", a);}

输入1234567890(超出指定大小),结果是

123456789

 输入1234,按enter(遇到换行符),结果是

1234

我们可以再看个例子啊

#include <stdio.h>
int main(void)
{char words[10];puts("Enter strings (empty line to quit):");while (fgets(words, 14, stdin) != NULL && (words[0] != '\n'))fputs(words, stdout);puts("Done.");
}

输入By the way,the gets() function, 结果是

有人就会有疑问了啊,这输入的东西不是超除了words的大小吗?那为什么还能正常打印?

实际上它确实超过了,但是这是循环!

程序中的fgets()一次读入 10 -1个字符(该例中为9个字符)。所以,一开始它只读入了“By the wa”,并储存为By the wa\0:接着fputs()打印该字符串,而且并未换行然后while循环进入下一轮迭代,fgets()继续从剩余的输入中读入数据,即读入“y,the ge”并储存为y,the ge\0接着fputs()在刚才打印字符串的这一行接着打印第2次读入的字符串。然后while 进入下一轮迭代,fgets()继续读取输入、fputs()打印字符串,这一过程循环进行,直到读入最后的“tion\n”fgets()将其储存为tion\n\0,fputs()打印该字符串,由于字符串中的\n,光标被移至下一行开始处。

保留换行符

需要注意的是,fgets()函数会保留输入流中的换行符,所以读取到的字符串可能包含换行符。如果你希望去除换行符,可以使用strtok()或者手动处理字符串

系统采用缓冲的IO,这意味着用户在按下enter键之前,输入都会被存在临时存储区(缓冲区)。

这点与gets()不同,gets()会丢弃换行符。

我们可以先借用puts()函数的特性:自动在字符串末尾加换行符

我们可以验证一下

#include <stdio.h>
int main(void)
{char words[14];puts("请输入:");fgets(words, 14, stdin);printf("见证奇迹的时刻:\n");puts(words);printf("sjajj");}

结果是: 

apple pie,比fgets()读入的整行输入短,因此,apple pie\n\0被储存在数组中(因为fgets()会自动存储换行符)。当puts()显示该字符串时又在末尾添加了换行符,调用puts()apple pie\n\0里的换行符起作用将光标移动到下一行,但是puts自动在字符串末尾添加换行符,所以光标再次移动到下一行。因此apple pie下面有一行空行。

系统使用缓冲的I/O。这意味着用户在按下Return键之前,输入都被储存在临时存储区(即,缓冲区)中。按下Enter键就在输入中增加了一个换行符,并把整行输入发送给fgets()。对于输出,fputs()把字符发送给另一个缓冲区,当发送换行符时,缓冲区中的内容被发送至屏幕上。

fgets()储存换行符有好处也有坏处。

坏处是你可能并不想把换行符储存在字符串中,这样的换行符会带来一些麻烦。

好处是对于储存的字符串而言,检查末尾是否有换行符可以判断是否读取了一整行。如果不是一整行,要妥善处理一行中剩下的字符。

处理掉换行符


首先,如何处理掉换行符?

一个方法是在已储存的字符串中查找换行符,并将其替换成空字符:

while (words[i] !='\n')// 假设\n在words中
i++;
words[i]='\0';


其次,如果仍有字符串留在输入行怎么办?

一个可行的办法是,如果目标数组装不下一整行输入,就丢弃那些多出的字符

while(getchar()!='\n')
contine;

gets_s()函数

C11标准新增的gets_s()函数也可代替gets()。该函数与gets()函数更接近,而且可以替换现有代码中的 gets()。但是,它是stdio.h.输入/输出函数系列中的可选扩展,所以支持C11的编译器也不一定支持它。

在C语言中,gets_s()函数用于读取用户输入的字符串。gets_s()函数的声明如下:

char *gets_s(char *str, rsize_t n);

其中,str是指向字符数组的指针,用于存储读取到的字符串;n表示字符数组的大小。

gets_s()函数会读取用户输入的字符串,并将其存储到指定的字符数组中,直到读取到换行符或数组大小的限制。读取到的字符串将包含换行符,且以\0字符结尾。

需要注意的是,gets_s()函数是C11中引入的安全版本的函数,主要解决了gets()函数的缓冲区溢出问题。

C11新增的gets_s()函数(可选)和fgets()类似,用一个参数限制读入的字符数。


特性

  1. gets_s()只从标准输入中读取数据,所以不需要第3个参数。
  2. 如果gets_s()读到换行符,会丢弃它而不是储存它。
  3. 如果gets_s()读到最大字符数都没有读到换行符,会执行以下几步。首先把目标数组中的首字符设置为空字符,读取并丢弃随后的输入直至读到换行符或文件结尾,然后返回空指针。接着,调用依赖实现的“处理函数”(或你选择的其他函数),可能会中止或退出程序。

第2个特性说明,只要输入行未超过最大字符数,gets_s()和gets()几乎一样,完全可以用gets_s()换gets()。第3个特性说明,要使用这个函数还需要进一步学习。

三种输入方式的选择

我们来比较一下gets()、fgets()和gets_s()的适用性。

如果目标存储区装得下输入行,3个函数都没问题。但是fgets()会保留输入末尾的换行符作为字符串的一部分,要编写额外的代码将其替换成字符。

如果输入行太长会怎样?

使用gets()不安全,它会擦写现有数据,存在安全隐患。gets_s()函数很全,但是,如果并不希望程序中止或退出,就要知道如何编写特殊的“处理函数”。另外,如果打算让程继续运行,gets_s(会丢弃该输入行的其余字符,无论你是否需要。由此可见,当输入太长,超过数组容纳的字符数时,fgets()函数最容易使用,而且可以选择不同的处理方式。如果要让程序继续使用输中超出的字符,可以参考程序清单11.8中的处理方法。
所以,当输入与预期不符时,gets_s()完全没有fgets()函数方便、灵活。也许这也是gets s二民的因之一。鉴于此,fgets()通常是处理类似情况的最佳选择。
 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/712834.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ai图生文的软件!分享4个受欢迎的!

在数字化时代&#xff0c;随着人工智能技术的飞速发展&#xff0c;AI图生文软件已经成为自媒体人、创作者和广告从业者手中的得力助手。这些软件能够将静态的图片转化为生动的文字&#xff0c;为图片注入灵魂&#xff0c;让观者仿佛置身于画面之中。今天&#xff0c;就让我们一…

LabVIEW和Python开发微细车削控制系统

LabVIEW和Python开发微细车削控制系统 为满足现代精密加工的需求&#xff0c;开发了一套基于LabVIEW和Python的微细车削控制系统。该系统通过模块化设计&#xff0c;实现了高精度的加工控制和G代码的自动生成&#xff0c;有效提高了微细车削加工的自动化水平和编程效率。 项目…

1950-2022年各省逐年平均降水量数据

1950-2022年各省逐年平均降水量数据 1、时间&#xff1a;1950-2022年 2、指标&#xff1a;省逐年平均降水量 3、范围&#xff1a;33省&#xff08;不含澳门&#xff09; 4、指标解释&#xff1a;逐年平均降水数据是指当年的日降水量的年平均值&#xff0c;不是累计值&#…

ONLYOFFICE 桌面编辑器 v8.0 更新内容详细攻略

文章目录 引言PDF 表单RTL 支持电子表格中的新增功能Moodle 集成用密码保护 PDF 文件从“开始”菜单快速创建文档本地界面主题下载安装桌面编辑工具总结 引言 官网链接&#xff1a; ONLYOFFICE 官方网址 ONLYOFFICE 桌面编辑器是一款免费的文档处理软件&#xff0c;适用于 Li…

uniapp实现-审批流程效果

一、实现思路 需要要定义一个变量, 记录当前激活的步骤。通过数组的长度来循环数据&#xff0c;如果有就采用3元一次进行选择。 把循环里面的变量【name、status、time】, 全部替换为取出的那一项的值。然后继续下一次循环。 虚拟的数据都是请求来的, 组装为好渲染的格式。 二…

【打工日常】使用docker部署在线PDF工具

一、Stirling-PDF介绍 Stirling-PDF是一款功能强大的本地托管的基于 Web 的 PDF 操作工具&#xff0c;使用 docker部署。该自托管 Web 应用程序最初是由ChatGPT全权制作的&#xff0c;现已发展到包含广泛的功能来处理您的所有 PDF 需求。允许对 PDF 文件执行各种操作&#xff0…

基于session注册JAva篇springboot

springboot3全家桶&#xff0c;数据库 &#xff1a;redis&#xff0c;mysql 背景环境&#xff1a;邮箱验证码&#xff0c;验证注册 流程&#xff1a;先通过邮箱验证&#xff0c;发送验证码&#xff0c;将获取到的session和验证码&#xff0c;存入redis里&#xff08;发送邮箱…

【leetcode】链表的回文结构

大家好&#xff0c;我是苏貝&#xff0c;本篇博客带大家刷题&#xff0c;如果你觉得我写的还不错的话&#xff0c;可以给我一个赞&#x1f44d;吗&#xff0c;感谢❤️ 点击查看题目 思路: 1.找中间节点 找中间节点的方法在下面这个博文中详细提过 【点击进入&#xff1a;【l…

鸿蒙Harmony应用开发—ArkTS声明式开发(通用属性:布局约束)

通过组件的宽高比和显示优先级约束组件显示效果。 说明&#xff1a; 从API Version 7开始支持。后续版本如有新增内容&#xff0c;则采用上角标单独标记该内容的起始版本。 aspectRatio aspectRatio(value: number) 指定当前组件的宽高比。 卡片能力&#xff1a; 从API vers…

浅谈 Linux 孤儿进程和僵尸进程

文章目录 前言孤儿进程僵尸进程 前言 本文介绍 Linux 中的 孤儿进程 和 僵尸进程。 孤儿进程 在 Linux 中&#xff0c;就是父进程已经结束了&#xff0c;但是子进程还在运行&#xff0c;这个子进程就被称作 孤儿进程。 需要注意两点&#xff1a; 孤儿进程最终会进入孤儿院…

软考-计算题

1.二维矩阵转换成一维矩阵 2.算术表达式&#xff1a; 3.计算完成项目的最少时间&#xff1a;之前和的max&#xff08;必须之前的所有环节都完成&#xff09; 松弛时间&#xff1a;最晚开始时间-最早开始时间 最早&#xff1a;之前环节都完成的和的max 最晚&#xff1a;总时间…

黑猫的牌面

解法&#xff1a; 桶 #include <iostream> #include <vector> #include <algorithm> using namespace std; #define endl \nint main() {ios::sync_with_stdio(false);cin.tie(0); cout.tie(0);vector<int> tong(1001);int t 4;int k, pai;long lon…

LeetCode 每日一题 树合集 Day 16 - 27

终于是开学了&#xff0c;想了想每日一更频率太高&#xff0c;以后每周更新一周的每日一题。 103. 二叉树的锯齿形层序遍历 给你二叉树的根节点 root &#xff0c;返回其节点值的 锯齿形层序遍历 。&#xff08;即先从左往右&#xff0c;再从右往左进行下一层遍历&#xff0c…

嵌入式开发——面试题操作系统(调度算法)

linux7种进程调度算法 1&#xff1a;先来先服务&#xff08;FCFS&#xff09;调度算法 原理&#xff1a;按照进程进入就绪队列的先后次序进行选择。对于进程调度来说&#xff0c;一旦一个进程得到处理机会&#xff0c;它就一直运行下去&#xff0c;直到该进程完成任务或者因等…

阿里云降价,这泼天的富贵你接不接?附云服务器价格表

阿里云能处&#xff0c;关键时刻ta真降价啊&#xff01;2024新年伊始阿里云带头降价了&#xff0c;不只是云服务器&#xff0c;云数据库和存储产品都降价&#xff0c;阿里云新老用户均可购买99元服务器、199元服务器&#xff0c;续费不涨价&#xff0c;阿里云百科aliyunbaike.c…

【力扣hot100】刷题笔记Day17

前言 今天竟然不用开组会&#xff01;天大的好消息&#xff0c;安心刷题了 46. 全排列 - 力扣&#xff08;LeetCode&#xff09; 回溯&#xff08;排列&#xff09; class Solution:def permute(self, nums: List[int]) -> List[List[int]]:# 回溯def backtrack():if len(…

关于游戏报错提示x3daudio1_7.dll丢失怎么修复?多个实测有效方法分享

x3daudio1_7.dll 是一个与 Microsoft DirectX 相关的重要动态链接库&#xff08;DLL&#xff09;文件&#xff0c;它主要服务于Windows操作系统下的多媒体和游戏应用程序。 一、以下是关于 x3daudio1_7.dll 文件的详细介绍 名称与位置&#xff1a; 文件名&#xff1a;x3daud…

探秘Python的Pipeline魔法

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站AI学习网站。 目录 前言 什么是Pipeline&#xff1f; Pipeline的基本用法 Pipeline的高级用法 1. 动态调参 2. 并行处理 3. 多输出 …

Spring底层源码分析

spring依赖注入底层原理解析 spring之bean对象生命周期步骤详情 流程&#xff1a; UserService.class —>推断构造方法—>普通对象----依赖注入------>初始化&#xff08;afterPropertiesSet方法&#xff09;------>初始化后&#xff08;AOP&#xff09;------…

Zabbix“专家坐诊”第231期问答

问题一 Q&#xff1a;用docker-compose部署zabbix&#xff0c;部署完后如果要修改zabbix的配置应该要改docker-compose文件里的环境变量吧&#xff1f;改了环境变量之后只能重建容器才能生效吗&#xff1f;能不能在不影响已经配好的那些监控项的情况下让新的环境变量生效&#…