C语言常见字符串函数 —— gets()、fgets()、gets_s()详解及其注意事项

目录

前言

gets()函数

1.描述

2.函数原型

3.参数

4.返回值

5.缺陷

fgets()函数

1.描述

2.函数原型

3.参数

4.返回值

 gets_s函数

1.描述

2.函数原型

3.参数

4.返回值

5.常见报错


前言

以下内容总结自《C primer plus》与 visual studio 2022语言参考库,仅供参考。

gets()函数

1.描述

属于C标准库<stdio.h>

gets()函数 会读取整行输入行,直至遇到换行符,然后丢弃换行符,储存其余字符,并在字符末尾添加一个空字符使其成为C字符串。常和 puts()函数 配对使用。

2.函数原型

char* gets(char* str)

3.参数

指向储存输入内容地址的指针。

4.返回值

成功读取输入会返回储存输入内容的地址,发生错误或未读取任何非 '\n'字符则返回NULL(空指针)。

5.缺陷

  • 当gets()函数的参数是有限大小的字符数组时,其唯一的参数是该字符数组首元素的地址(即指向该字符串首元素的指针),它无法检查数组的空间是否足够储存所有输入行。如果输入的字符串过长,会导致缓冲区溢出,多余的字符会超出指定的目标空间。

    如果超出的字符仅占用了尚未使用的内存,就不会立即出现问题;如果它们修改了程序中的其他数据,会导致程序异常或中止;或其他情况。

  • 例如:在UNIX系统中,会警告该程序试图访问未分配的内存。该函数的不安全行为和使用时的不严谨会造成安全隐患。



fgets()函数

1.描述

属于C标准库<stdio.h>

  • 读取指定数量的字符,如果该参数是 n,则读取 n-1 个字符,或读取到第一个换行符为止。超出的字符则保留在缓冲区中。

    (可选择性利用 getchar()循环清空缓冲区中的字符,避免影响下次输入)

//仅供参考
while (getchar() != '\n')
    continue;
  • 与 gets()不同的是 gets()会丢弃换行符替换为空字符,而 fgets()会储存在字符串中。
  • fgets()通过第2个参数限制读入的字符数来解决 gets()函数 的溢出问题。该函数专门设计用于处理文件输入。

2.函数原型

char* fgets(char* str, int n, FILE* stream)

3.参数

  1. str --- 指向储存输入内容地址的指针。
  2. n --- 要读取的最大字符数量(包括字符串末尾的空字符)。
  3. stream --- 指向FILE对象的指针,指明要读入的文件。如果从键盘中读入输入的数据,则以 stdin作为参数,该标准输入标识符定义在<stdio.h>中。

4.返回值

成功读取输入会返回与参数 str相同的地址,发生错误、读取到文件尾或未读取到任何非 '\n'字符则返回NULL(空指针)。 



 gets_s函数

1.描述

由C11标准新增。属于<stdio.h>输入/输出函数系列中的可选扩展,支持C11的编译器也不一定支持它。

  • 读取指定数量的字符,如果参数是 n,则读取 n - 1个字符,如果读取到换行符会像gets()一样丢弃换行符,替换成空字符。
  • 与 fgets()类似,该函数会用一个参数限制读入的字符数,但 fgets()限制的是函数读取的字符数量,而 gets_s()限制的是缓存区大小。
  • 与 fgets()相比 gets_s()只从标准输入(stdin)中读取数据,所以不需要第三个参数。

2.函数原型

char* gets_s(char* str, rsize_t n)

3.参数

  1. str --- 指向储存输入内容地址的指针。

  2. n --- 缓冲区大小。

4.返回值

 如果成功会返回与参数 str相同的地址,发生错误或未读取到任何非 '\n'字符则返回NULL(空指针)。 

5.常见报错

  • 编译器支持性问题:gets_s()函数是由C11标准新增的函数,主要用于解决前文提到的安全性问题,因此并非所有编译器都支持gets_s()函数。
  • 缓冲区溢出问题:由于gets_s()函数的第二参数是限制缓冲区大小,当输入的字符数超过了缓冲区大小时,就会导致缓冲区溢出。

以下为 Visual studio 2022环境下缓冲区溢出现象

#include <stdio.h>

int main()
{
	char arr[10];
	gets_s(arr, 5);
	puts(arr);
	return 0;
}

<think>嗯,用户问的是C语言中的gets函数的使用说明和替代方案。首先,我需要回忆一下gets函数的基本用法和存在的问题。记得gets函数是用来从标准输入读取字符串的,直到遇到换行符或EOF为止。不过,这个函数有个很大的问题,就是它不检查缓冲区的大小,容易导致缓冲区溢出,这可能被恶意利用,所以后来被弃用了。 接下来,我需要整理gets函数的典型用法。例如,用户声明一个字符数组,比如char str[81],然后用gets(str)来读取输入。这时候,用户输入的内容会被存储到str中,包括空格,因为gets会读取整行,直到换行符。这点和scanf不同,scanf遇到空格就会停止,所以gets适合需要包含空格的字符串输入。 然后,关于gets函数的问题,主要有三点。首先是缓冲区溢出风险,因为gets无法限制输入长度,如果用户输入超过数组长度,就会覆盖其他内存区域,导致程序崩溃或安全漏洞。其次是C11标准已经移除了gets函数,编译器可能会给出警告或错误。再者,虽然有些编译器支持gets_s作为替代,但这不是所有环境都可用,可移植性差。 接下来是替代方案。第一个替代是使用fgets函数,它需要指定最大读取长度和输入流,比如fgets(str, sizeof(str), stdin),这样能有效防止溢出。不过要注意fgets会保留换行符,可能需要处理掉。第二个替代方案是用C++的getline,但用户的问题是关于C语言的,所以可能更推荐fgets。第三个是gets_s,不过如前所述,可移植性可能有问题。 然后需要引用相关的资料。根据引用[1],gets不安全,建议用fgets或C++的getline。引用[2]提到gets可以输入空格,而scanf不行。引用[3]说明gets_s是C11的可选扩展,不一定所有编译器支持。引用[4]给出了gets的示例代码。 用户可能需要知道具体如何使用这些替代函数,比如fgets的使用示例,如何处理换行符。另外,安全问题需要强调,尤其是在实际项目中使用gets的风险。可能还需要比较不同替代方案的优缺点,帮助用户根据具体情况选择。 最后,生成相关问题时要考虑用户可能的后续疑问,比如如何安全输入字符串,不同函数的区别,如何处理输入中的换行符,或者为什么gets被弃用。确保问题覆盖用户可能关心的各个方面,帮助他们深入理解。</think>### C语言gets函数使用说明及替代方案 #### 一、gets函数基础用法 1. **功能**:从标准输入(stdin)读取字符串,直到遇到换行符或文件结束符(EOF),并将换行符替换为`\0`。例如: ```c char str[81]; gets(str); // 输入"Hello World"会完整存储到str中 ``` 注意:`gets()`会读取输入中的空格,例如输入`"C programming"`会被完整接收[^2]。 2. **典型问题**: - **缓冲区溢出**:若输入数据长度超过数组容量,例如声明`char str[5]`但输入`"abcdef"`,会导致内存越界,可能引发程序崩溃或安全漏洞[^1][^3]。 - **标准支持问题**:C11标准已移除`gets()`,编译器会提示警告(如GCC的`warning: implicit declaration of 'gets'`)。 #### 二、替代方案 1. **fgets函数(推荐)** - **语法**:`char *fgets(char *str, int n, FILE *stream)` - **优点**:限制最大读取长度,防止溢出。例如: ```c char buffer[10]; fgets(buffer, sizeof(buffer), stdin); // 最多读取9个字符 ``` - **注意**:`fgets()`会保留换行符,需手动处理: ```c buffer[strcspn(buffer, "\n")] = '\0'; // 删除换行符 ``` 2. **C++的getline(跨语言方案)** 适用于混合C/C++项目: ```cpp #include <string> std::string input; std::getline(std::cin, input); // 自动处理动态内存 ``` 3. **gets_s函数(条件支持)** C11新增的可选扩展: ```c char buf[10]; gets_s(buf, sizeof(buf)); // 限制输入长度 ``` 注意:部分编译器(如MinGW)可能不支持此函数。 #### 三、对比总结 | 函数 | 安全性 | 换行符处理 | 标准支持 | |-----------|--------|------------|----------------| | `gets` | 不安全 | 自动删除 | C99及更早 | | `fgets` | 安全 | 保留 | 所有C/C++标准 | | `getline` | 安全 | 自动删除 | C++标准 | | `gets_s` | 安全 | 自动删除 | C11(可选) |
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值