C语言文件操作详解(上)(二)

简介: 本文详细介绍了C语言中的文件操作,包括fopen()、fclose()、fread()、fwrite()、fgetc()、fputc()、fgets()、fputs()、fscanf()和fprintf()等函数的使用。这些函数分别用于文件的打开、关闭、读取、写入和格式化输入输出。文章还通过示例代码解释了如何读取和写入字符串、结构体等数据,并提到了标准输入流(stdin)、标准输出流(stdout)和标准错误流(stderr)。

C语言文件操作详解(上)(一)+https://developer.aliyun.com/article/1519328?spm=a2c6h.13148508.setting.14.25634f0e6POlmr


5.fgets()函数的使用


(1)fgets()从文件中读取num-1个字符的情况


下列代码中,arr数组用于接收读到的字符。为了观察更清晰,我们预先在其中放入10个#号。


fgets(arr,5,pf);语句意思是从pf指向的文件中读取5个字符,存入arr中。新存入的字符会覆盖arr中原有的字符。



int main()
{
  //打开文件
  FILE* pf = fopen("test.txt", "r");
  if (NULL == pf)
  {
    perror("fopen");
    return 1;
  }
 
  //读文件-一行一行读
  //10个#
  char arr[] = "##########";
  fgets(arr, 5, pf);
  printf("%s", arr);
  
  //关闭文件
  fclose(pf);
  pf = NULL;
 
  return 0;
}


在代码运行前,test.txt文件中的内容如下:

abcdefg\nhijklmnopqrstuvwxyz



运行后,屏幕上显示arr数组内的内容:



如图,虽然我们要输入的num为5,但实际只显示出4个字符。通过调试我们可以发现,是因为系统在读入的时候,自动将在第4个字符后添加了一个'\0':



第5个字符为'\0'


因此,当num小于等于该行字符数时,实际存入arr的是该行前(num-1)个字符,最后一个字符为系统自动追加的'\0'。 而若令要存入的num数大于该行字符总数,则该行字符会被全部读取,包括末尾的'\n'。并且在读完后,还是会在末尾追加'\0'




一行共8个字符,但会另外追加一个'\0'在arr数组中


6.fputs()    文本行输出函数


fputs()函数的API文档:fputs - C++ Reference


函数原型


int fputs ( const char * str, FILE * stream );


C 库函数int fputs(const char *str, FILE *stream)把字符串写入到指定的文件流 stream 中,但不包括空字符。


参数和返回值

  • str -- 数组,包含了要写入的以空字符终止的字符序列。
  • stream -- 这是指向 FILE 对象的指针,该 FILE 对象标识了要被写入字符串的文件流。
  • 返回值 --若成功,返回一个非负值;若发生错误,则返回 EOF。


7.fputs()的使用


打印两行,注意要在字符串中加入换行符'\n',否则并不能实现在文件中显示两行的效果。


int main()
{
  //打开文件
  FILE* pf = fopen("test.txt", "w");
  if (NULL == pf)
  {
    perror("fopen");
    return 1;
  }
 
  //写文件-一行一行写
  fputs("hello\n", pf);
  fputs("world!!!!\n", pf);
 
  //关闭文件
  fclose(pf);
  pf = NULL;
 
  return 0;
}






8.fscanf()    格式化输入函数


fscanf()函数的API文档:fscanf - C++ Reference


函数原型

int fscanf ( FILE * stream, const char * format, ... );


C 库函数int fscanf(FILE *stream, const char *format, ...)  从文件流 stream 读取格式化输入。


返回值


  • 如果成功,该函数返回成功匹配和赋值的个数。如果到达文件末尾或发生读错误,则返回 EOF。


该函数的用法与scanf函数一致,只是fscanf()函数在参数部分多了一个文件指针用于标识要输入的文件流。


9.fprintf()    格式化输出函数


fprintf()函数的API文档:fprintf - C++ Reference


函数原型


int fprintf ( FILE * stream, const char * format, ... );


该函数的用法与printf函数也一致,只是fprintf函数在参数部分多了一个文件指针用于标识要输出到的文件流。


10.fscanf()与fprintf()函数的使用


(1)读写一个结构体中的数据


先创建一个结构体的数据:


struct S
{
  char name[20];
  int age;
  float score;
};

把结构体中的一些数据写到文件中:

int main()
{
  struct S s = { "zhangsan", 20, 95.5f };
  //把s中的数据写到文件中
  FILE* pf = fopen("test.txt", "w");
  if (NULL == pf)
  {
    perror("fopen");
    return 1;
  }
 
  //写文件
  fprintf(pf, "%s %d %.1f", s.name, s.age, s.score);
 
  fclose(pf);
  pf = NULL;
  return 0;
}


运行后效果如下:



2)读取一个结构体的数据


依旧是上述结构体,在写入数据后,我们再将它读取到内存中:



struct S
{
  char name[20];
  int age;
  float score;
};
 
int main()
{
  struct S s = {0};
  //把s中的数据写到文件中
  FILE* pf = fopen("test.txt", "r");
  if (NULL == pf)
  {
    perror("fopen");
    return 1;
  }
  //读文件
  fscanf(pf, "%s %d %f", s.name, &(s.age), &(s.score));
 
  printf("%s %d %f\n", s.name, s.age, s.score);
 
  fclose(pf);
  pf = NULL;
  return 0;
}


此时在控制台中也能成功打印出文件内容:



11.fread()    二进制输入


fread()函数的API文档:fread - C++ Reference


函数原型

size_t fread ( void * ptr, size_t size, size_t count, FILE * stream );


从给定文件流stream读取数据到 ptr 所指向的数组中。


参数


  • ptr --指向带有最小尺寸 size*nmemb 字节内存块的指针。
  • size -- 要读取的每个元素的大小,以字节为单位。
  • nmemb-- 元素个数,每个元素的大小为 size 字节。
  • stream  -- 指向 FILE 对象的指针,该 FILE 对象指定了一个输入文件流。


2.fwrite()   二进制输出


fwrite()函数的API文档:fwrite - C++ Reference


函数原型

size_t fwrite ( const void * ptr, size_t size, size_t count, FILE * stream );


ptr 所指向的数组中的数据写入到给定文件流 stream  中。


参数


  • ptr -- 这是指向要被写入的元素数组的指针。
  • size -- 这是要被写入的每个元素的大小,以字节为单位。
  • nmemb -- 这是元素的个数,每个元素的大小为 size 字节。
  • nmemb -- 这是指向 FILE 对象的指针,该 FILE 对象指定了一个输出文件流。


13.fread()与fwrite()的使用


(1)二进制的写文件


写一个结构体数据到文件中。注意文件打开模式为"wb"。


struct S
{
  char name[20];
  int age;
  float score;
};
 
int main()
{
  struct S s = { "zhangsan", 20, 95.5f };
  //把s中的数据写到文件中
  FILE*pf = fopen("test.txt", "wb");
  if (NULL == pf)
  {
    perror("fopen");
    return 1;
  }
  //二进制的写文件
  fwrite(&s, sizeof(s), 1, pf);
 
  fclose(pf);
  pf = NULL;
  return 0;
}


(2)二进制的读文件

struct S
{
  char name[20];
  int age;
  float score;
};
 
int main()
{
  struct S s = {0};
  //把s中的数据写到文件中
  FILE* pf = fopen("test.txt", "rb");
  if (NULL == pf)
  {
    perror("fopen");
    return 1;
  }
  //二进制的读文件
  fread(&s, sizeof(s), 1, pf);
 
  printf("%s %d %f\n", s.name, s.age, s.score);
 
  fclose(pf);
  pf = NULL;
  return 0;
}


五、输入输出流的补充


对于任何一个C程序,只要运行起来就默认打开三个流:


stdin -- 标准输入流 -- 键盘


stdout -- 标准输出流 -- 屏幕


stderr -- 标准错误流 -- 屏幕


上面介绍到,除了fread()与fwrite()适用于文件流以外,其余的顺序读写函数适用于所有输入流。因此,它们也可以起到scanf()与printf()的作用,如:


fgetc(stdin);
fputc(ch,stdout);


在需要填入一个文件指针的地方,将参数换为stdin或stdout即可。

相关文章
|
26天前
|
存储 C语言
【c语言】玩转文件操作
本文介绍了C语言中文件操作的基础知识,包括文件的打开和关闭、文件的顺序读写、文件的随机读写以及文件读取结束的判定。详细讲解了`fopen`、`fclose`、`fseek`、`ftell`、`rewind`等函数的使用方法,并通过示例代码展示了如何进行文件的读写操作。最后,还介绍了如何判断文件读取结束的原因,帮助读者更好地理解和应用文件操作技术。
34 2
|
1月前
|
存储 C语言
C语言文件操作(2)
【10月更文挑战第2天】
|
1月前
|
程序员 编译器 C语言
C语言底层知识------文件操作
本文详细介绍了文件操作的基本概念,包括文件的分类(程序文件和数据文件,其中着重于数据文件的文本文件和二进制文件),流的概念及其在C程序中的应用,以及标准输入输出流stdin、stdout和stderr的作用。作者通过示例展示了如何使用fopen、fclose和常见的读写函数如fgetc、fputc和fgets进行文件操作。
22 2
|
2月前
|
C语言
C语言——文件操作
本文介绍了文件的基本操作,包括文件的打开、关闭、读取和写入。使用`fopen`函数以不同模式(如“r”、“w”等)打开文件,并通过`fclose`关闭。文章详细解释了如何利用`fputc`、`fputs`及`fprintf`进行格式化写入,同时介绍了`fgetc`、`fgets`和`fscanf`用于文件内容的读取。此外,还涵盖了二进制文件的读写方法以及如何通过`fseek`、`ftell`和`rewind`实现文件的随机访问。
51 1
C语言——文件操作
|
1月前
|
存储 缓存 编译器
文件操作——C语言
文件操作——C语言
|
1月前
|
存储 C语言
简述C语言文件操作
简述C语言文件操作
11 0
|
1月前
|
存储 文件存储 C语言
深入C语言:文件操作实现局外影响程序
深入C语言:文件操作实现局外影响程序
|
1月前
|
存储 程序员 编译器
C语言文件操作(1)
【10月更文挑战第1天】
|
1月前
|
存储 C语言
C语言的文件操作
C语言的文件操作
21 0
|
1月前
|
存储 移动开发 Unix
C 语言文件操作详解
C 语言文件操作详解