掘根宝典之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()通常是处理类似情况的最佳选择。
 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

软件实例,佳易王账单账本记账汇总统计管理系统软件教程

软件实例&#xff0c;佳易王账单账本记账汇总统计管理系统软件教程 一、前言 以下软件程序教程 以 佳易王账单记账汇总统计管理系统软件V17.0为例说明 软件文件下载可以点击最下方官网卡片——软件下载——试用版软件下载 账单可以记录 1、收入明细 2、支出明细 3、客户…

信息安全是什么

信息安全&#xff0c;也称为信息安全或数据安全&#xff0c;是防止未经授权的访问、更改、中断和破坏信息。 信息安全本身包括的范围很大&#xff0c;大到国家军事政治等机密安全&#xff0c;小范围的当然还包括如防范商业企业机密泄露&#xff0c;防范青少年对不良信息的浏览…

Excel中筛选合并单元格后,只显示第一行怎么办?

Excel中筛选合并单元格后,只显示第一行怎么办? 参考链接:https://baijiahao.baidu.com/s?id=1736773058549439034&wfr=spider&for=pc 我们日常的Excel数据在展示的时候为了数据的清晰和美观往往部分相同的单元格进行合并,但是合并之后在筛选时会发现结果会显示异…

Vanna-ai -基于RAG的TextToSql实现方案

官方连接&#xff1a;Vanna.AI - Personalized AI SQL Agent 1.背景 基于大模型的TextToSql的关键为给大模型提供正确有效的数据库信息及问题&#xff0c;以提升大模型生成sql的正确率。database_info question形成prompt&#xff0c;但是实际中通常会遇到一个问题&#xff…

【前端】Vite打包页面简单部署到GitHub上

创建仓库---->上传代码---->设置 注意点已经打上箭头,代码我传到的是test分支 vite打包的配置如图&#xff0c;base是仓库名称&#xff0c;docs是build后生成的打包目录。 上传到GitHub就自动部署了 访问就是第一张图里的一串地址&#xff0c;这种方式比较方便吧

Claude 3 模型列表

claude-3-opus-20240229 这个模型就好

Midjourney入门:AI绘画真的能替代人类的丹青妙笔吗?

名人说&#xff1a;一花独放不是春&#xff0c;百花齐放花满园。——《增广贤文》 作者&#xff1a;Code_流苏(CSDN)&#xff08;一个喜欢古诗词和编程的Coder&#x1f60a;&#xff09; 目录 一、简要介绍1、Midjourney2、使用方法 二、绘画1、动物类2、风景类3、动漫类4、艺…

CNAN知识图谱辅助推荐系统

CNAN知识图谱辅助推荐系统 文章介绍了一个基于KG的推荐系统模型&#xff0c;代码也已开源&#xff0c;可以看出主要follow了KGNN-LS 。算法流程大致如下&#xff1a; 1. 算法介绍 算法除去attention机制外&#xff0c;主要的思想在于&#xff1a;user由交互过的item来表示、i…

VMvare安装17安装centos8教程

阿里镜像站&#xff1a;https://mirrors.aliyun.com/centos centos-8-isos-x86_64安装包下载_开源镜像站-阿里云 https://mirrors.aliyun.com/centos/8/isos/x86_64/CentOS-8.5.2111-x86_64-dvd1.iso 将上面的链接复制到迅雷进行高速下载 vmvare安装配置教程安装教程 CentO…

Vue3 五天速成

文章目录 day 11. 创建vue3工程3. 响应式数据4. 计算属性 day 25. watch 监视6. watchEffect7. 标签的ref属性 day 38. 回顾TS中的接口_泛型_自定义类型9. props的使用10. 生命周期11. 自定义Hooks12. 路由 基本切换效果13. 路由 两个注意点14. 路由 路由器的工作模式15. 路由 …

【C++】类和对象终篇

个人主页 &#xff1a; zxctscl 文章封面来自&#xff1a;艺术家–贤海林 如有转载请先通知 文章目录 1. 前言2. 友元2.1 友元函数2.2 友元类 3. 内部类4. 匿名对象5. 拷贝对象时的一些编译器优化6. 再次理解类和对象 1. 前言 在上一篇博客中提到了类和对象中的构造函数与stat…

网盘拉新平台,如何授权对接“星子助推”?

找到“星子助推”&#xff1a;首先&#xff0c;找到“星子助推”这个授权渠道。他们是网盘服务提供商的合作伙伴&#xff0c;为你提供机会。注册并申请授权&#xff1a;在“星子助推”的平台上注册&#xff0c;并同时申请授权。填写邀请码8x25k&#xff0c;提交申请。获得授权并…

云服务器2核4G能支持多少人同时访问?2核4G5M并发量评测

腾讯云轻量应用服务器2核4G5M配置一年优惠价165元、252元15个月、三年756元&#xff0c;100%CPU性能&#xff0c;5M带宽下载速度640KB/秒&#xff0c;60GB SSD系统盘&#xff0c;月流量500GB&#xff0c;折合每天16.6GB流量&#xff0c;超出月流量包的流量按照0.8元每GB的价格支…

#QT(网络编程-UDP)

1.IDE&#xff1a;QTCreator 2.实验&#xff1a;UDP 不分客户端和服务端 3.记录 &#xff08;1&#xff09;做一个UI界面 &#xff08;2&#xff09;编写open按钮代码进行测试&#xff08;用网络调试助手测试&#xff09; &#xff08;3&#xff09;完善其他功能测试 4.代码 …

设计模式——2_3 迭代器(Iterator)

生活就像一颗巧克力&#xff0c;你永远不知道下一颗是什么味道 ——《阿甘正传》 文章目录 定义图纸一个例子&#xff1a;假如你的供应商提供了不同类型的返回值单独的遍历流程实现 碎碎念如果读写同时进行会发生啥&#xff1f;外部迭代和内部迭代迭代器和其他模式迭代器和组合…

AI从截图直接生成代码、前端程序员的福音

简介 项目可以将任何屏幕截图或设计转换为干净的代码&#xff08;支持大多数框架&#xff09;。来自领先公司的开发人员和设计师使用的排名第一的工具。完全开源&#xff0c;在 GitHub 上拥有超过 35,000 颗星。非常受欢迎。 各位小伙伴们感觉有帮助的&#xff0c;可以收藏一…

【促销定价】背后的算法技术3-数据挖掘分析

【促销定价】背后的算法技术3-数据挖掘分析 01 整体分析1&#xff09;整体概览2&#xff09;类别型特征概览3&#xff09;数值型特征概览 02 聚合分析1&#xff09;天维度2&#xff09;品维度3&#xff09;价格维度4&#xff09;数量维度 03 相关分析1&#xff09;1级品类2&…

指针中的回调函数与qsort的深度理解与模拟

今天给大家在更新一下指针类型的知识&#xff0c;这里讲到了一个库函数sqort&#xff0c;以及回调函数的理解。 望喜欢 目录 回调函数 qsort函数 qsort模拟实现 回调函数 回调函数就是⼀个通过函数指针调用的函数。 如果你把函数的指针&#xff08;地址&#xff09;作为参数…

利用文件实现进程间共享数据

概述 文件可以存储任何非结构化字节序列&#xff0c;这个比较简单&#xff0c;就一个写一个读&#xff1b;学习到此&#xff0c;留个记录&#xff0c;以后可以直接抄代码&#xff0c;哈哈 Demo代码 #include <fstream> #include <iostream> #include <thread&…

CMIP6数据处理方法与典型案例分析

气候变化对农业、生态系统、社会经济以及人类的生存与发展具有深远影响&#xff0c;是当前全球关注的核心议题之一。IPCC&#xff08;Intergovernmental Panel on Climate Change&#xff0c;政府间气候变化专门委员会&#xff09;的第六次评估报告明确&#xff1b;指出&#x…