fgets 之用法

本文详细介绍了fgets函数的使用方法,包括其参数含义、返回值及实际应用案例。通过具体示例展示了如何利用fgets从文件中读取字符串,并对其进行了深入解析。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

今天在看ObjectiveC,看到了一个fgets的例子,才发现对于fgets的理解不够透彻。

fgets 的使用方法:char *fgets(char *string, int n, FILE *stream) 

从文件stream中读取n-1个字符/一行(若一行不满n-1个),string接收字符串

 如果n <= 0,返回NULL

 如果n == 1,返回" ",也就是一个空串

 如果成功,返回值等于string, 也就是获得字符串的首地址

 如果出错,或者读到FILE的结尾,返回NULL.

下面看例子:


#include <stdio.h>
#include <string.h>

int main( )
{
 FILE* wordFile = fopen("words.txt","r");
 char word[100];
 
 while(fgets(word, 100, wordFile))
 {
  word[strlen(word)-1] = '/0';
  printf("%s is %d characters long/n", word, strlen(word));
 }
 fclose(wordFile);
 return (0);
}

words.txt的内容如下(共4行,第4行,为空):

apple trees
many books on the desk
have a cup of water

运行后,输出结果如下:

apple trees is 11 characters long

many books on the desk is 22 characters long

have a cup of water is 19 charcters long

从上面的例子看出,若一行不满100个字符时,fgets可以从文件中读取一行。这点,以前,没有注意到。


原文来自:http://blog.youkuaiyun.com/lly66/archive/2008/10/19/3105779.aspx

fgets 函数的使用

fgets 既可以读文件,又可以读标准输入,而且可以防止溢出。但是它只能输入字符串(且能读到回车符/n),故而用scanf语句的较多。scanf语句可以输入各种格式的数据,其功能较为强大。

fgets 的使用方法:char *fgets(char *string, int n, FILE *stream) 

从文件stream中读取n-1个字符/一行(若一行不满n-1个),string接收字符串

 如果n <= 0,返回NULL

 如果n == 1,返回" ",也就是一个空串

 如果成功,返回值等于string, 也就是获得字符串的首地址

 如果出错,或者读到FILE的结尾,返回NULL

//通过while循环一行行取,读到文件末尾就是NULL了 ----读取整个文件
#include <stdio.h>

void main( void )
{
FILE *stream;
char line[100];

if( (stream = fopen( "file.txt", "r" )) != NULL )
{
while( fgets( line, 100, stream ) != NULL)
printf( "%s", line);
fclose( stream );
}
}

以下是fgets这个函数的实现:

/***
*fgets.c - get string from a file
*
* Copyright (c) Microsoft Corporation. All rights reserved.
*
*Purpose:
* defines fgets() - read a string from a file
*
*******************************************************************************/

#include <cruntime.h>
#include <stdio.h>
#include <dbgint.h>
#include <file2.h>
#include <internal.h>
#include <mtdll.h>
#include <tchar.h>

/***
*char *fgets(string, count, stream) - input string from a stream
*
*Purpose:
* get a string, up to count-1 chars or '/n', whichever comes first,
* append '/0' and put the whole thing into string. the '/n' IS included
* in the string. if count<=1 no input is requested. if EOF is found
* immediately, return NULL. if EOF found after chars read, let EOF
* finish the string as '/n' would.
*
*Entry:
* char *string - pointer to place to store string
* int count - max characters to place at string (include /0)
* FILE *stream - stream to read from
*
*Exit:
* returns string with text read from file in it.
* if count <= 0 return NULL
* if count == 1 put null string in string
* returns NULL if error or end-of-file found immediately
*
*Exceptions:
*
*******************************************************************************/

_TSCHAR * __cdecl _fgetts (
_TSCHAR *string,
int count,
FILE *str
)
{
REG1 FILE *stream;
REG2 _TSCHAR *pointer = string;
_TSCHAR *retval = string;
int ch;

_VALIDATE_RETURN(( string != NULL ) || ( count == 0 ), EINVAL, NULL);
_VALIDATE_RETURN(( count >= 0 ), EINVAL, NULL);
_VALIDATE_RETURN(( str != NULL ), EINVAL, NULL);

if (count == 0)
{
return NULL;
}

/* The C Standard states the input buffer should remain
unchanged if EOF is encountered immediately. Hence we
do not blank out the input buffer here */

/* Init stream pointer */
stream = str;

_lock_str(stream);
__try {

#ifndef _UNICODE
_VALIDATE_STREAM_ANSI_SETRET(stream, EINVAL, retval, NULL);
#endif /* _UNICODE */
if(retval!=NULL)
{
while (--count)
{
if ((ch = _fgettc_nolock(stream)) == _TEOF)
{
if (pointer == string) {
retval=NULL;
goto done;
}

break;
}

if ((*pointer++ = (_TSCHAR)ch) == _T('/n'))
break;
}
*pointer = _T('/0');
}


/* Common return */
done:

; }
__finally {
_unlock_str(stream);
}

return(retval);
}

-----------------------------------------------------------------------------------------------------------------

另: fgetc 和 getchar 是一个一个字符的读取,

 

<think>好的,我现在需要回答用户关于C语言中fgets函数的使用教程和示例代码的问题。首先,我得回想一下自己对fgets函数的了解。fgets主要用于从文件或标准输入读取一行数据,相比gets函数更安全,因为它可以指定缓冲区大小,避免溢出。 接下来,用户需要教程和示例代码,所以应该分步骤说明。第一步可能需要介绍函数原型,参数含义。比如函数原型是char *fgets(char *str, int n, FILE *stream),其中str是存储数据的缓冲区,n是最大读取字符数,stream是文件指针。这里要注意n包括最后的空字符,所以实际读取的字符数是n-1。 然后,需要给出示例代码。比如从标准输入读取数据,使用stdin作为流。比如定义一个缓冲区char buffer[100],然后调用fgets(buffer, sizeof(buffer), stdin)。要提醒用户注意,fgets会读取换行符并存入缓冲区,这点和gets不同,gets会丢弃换行符。所以在处理字符串时可能需要去掉换行符。 另外,还要说明返回值。如果成功,返回指向str的指针,失败或到达文件末尾返回NULL。这时候需要检查返回值,避免后续处理出错。 然后,可能还需要比较fgets和scanf的区别,比如fgets读取整行,而scanf根据格式字符串读取,可能留下换行符在输入缓冲区中。所以对于行输入,fgets更合适。 还需要提到文件读取的示例,比如从文件中逐行读取,直到结束。这时候可以用fopen打开文件,循环使用fgets读取每一行,处理后再关闭文件。 还要注意一些常见问题,比如缓冲区大小不足时的截断处理,换行符的处理方法,比如用strcspn(buffer, "\n")来找到换行符的位置并替换为'\0'。 最后,确保示例代码正确,没有语法错误,并且注释清晰。可能需要提供两个例子,一个从标准输入读取,另一个从文件读取,这样用户能全面了解用法。 现在,整理这些思路,分步骤写出教程,并附上示例代码。同时,生成相关问题,比如如何避免换行符、处理文件结束等。还要检查是否有引用需要添加,比如用户提供的引用[1]中的示例,可能需要引用说明。</think>### C语言fgets函数使用教程 #### 一、函数原型 `char *fgets(char *str, int n, FILE *stream);` 参数说明: - `str`:字符数组指针,用于存储读取的数据 - `n`:最大读取字符数(包含结尾的空字符`\0`) - `stream`:输入流(如`stdin`或文件指针) #### 二、核心特性 1. 安全读取:自动限制输入长度,防止缓冲区溢出[^1] 2. 保留换行符:若输入包含换行符,会将其存入缓冲区 3. 文件/标准输入通用:支持从键盘输入或文件读取 #### 三、使用步骤 1. 声明足够大的字符数组作为缓冲区 2. 调用`fgets`读取数据 3. 检查返回值 4. 处理换行符(可选) #### 四、基础示例(标准输入) ```c #include <stdio.h> #include <string.h> int main() { char buffer[100]; printf("请输入文本:"); // 读取输入(含换行符) if (fgets(buffer, sizeof(buffer), stdin)) { // 去除换行符 buffer[strcspn(buffer, "\n")] = '\0'; printf("输入内容:%s\n", buffer); } else { printf("读取失败!"); } return 0; } ``` #### 五、文件读取示例 ```c #include <stdio.h> int main() { FILE *fp = fopen("data.txt", "r"); if (!fp) { perror("文件打开失败"); return 1; } char line[256]; while (fgets(line, sizeof(line), fp)) { printf("行内容:%s", line); } fclose(fp); return 0; } ``` #### 六、关键注意事项 1. **返回值检查**:始终验证返回值是否为`NULL` 2. **缓冲区管理**:确保分配足够空间(包含`\n`和`\0`) 3. **输入残留**:混合使用`scanf`和`fgets`时需清空输入缓冲区 4. **跨平台差异**:Windows换行符为`\r\n`,Linux为`\n` #### 七、常见问题解决方案 **问题1:输入内容包含换行符** ```c // 使用字符串处理函数移除 size_t len = strlen(buffer); if (len > 0 && buffer[len-1] == '\n') { buffer[len-1] = '\0'; } ``` **问题2:读取文件时判断结束** ```c while (fgets(line, sizeof(line), fp) != NULL) { // 处理内容 } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值