【C语言】:文件读写相关函数介绍

简介: 【C语言】:文件读写相关函数介绍

C语言文件读写相关函数介绍

一.什么是文件

磁盘上的文件就是文件。

但是在程序设计中,我们谈的文件有两种:程序文件,数据文件。

1.1 程序文件

包括源程序文件(后缀为.c),目标文件(windows环境后缀为.obj),可执行程序(windows环境后缀为.exe)。

1.2 数据文件

文件的内容不一定是程序,而是程序运行时读写的数据,比如程序运行时需要从中读取数据的文件,或者输出内容的文件。

本文讨论的是数据文件

1.3 文件名

一个文件要有唯一的文件标识,以便用户识别和引用。

文件名包括3部分:文件路径+文件名主干+文件后缀

例如:c:\code\test.txt

其中c:\code\是文件路径,test是文件主干名,.txt是文件后缀。

为了方便起见,文件标识常被称为文件名

二.文件的打开和关闭

2.1 文件指针

缓冲文件系统中,关键的概念是"文件类型指针",简称"文件指针"。

每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件的状态,文件当前的位置等)。这些信息是保存在一个结构体变量中的。该结构体类型(该结构体如何实现我们不做关注)是系统声明的,取名FILE.

每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中信息,使用者不必关心细节。

下面我们可以创建一个FILE* 的指针变量:

FILE* PF;//文件指针变量

定义pf是一个指向FILE类型数据的指针变量。可以是pf指向某个文件的文件信息区(是一个结构体变量)。通过文件信息区中的信息就能够访问该文件。也就是说,通过文件指针变量能够找到与它关联的文件。

2.2 文件的打开和关闭函数

以下2个函数引用的头文件均为"stdio.h"

文件的打开函数fopen

参数是:fopen(文件名,文件的打开模式)

其中常用打开模式有:只读"r",只写"w",追加"a",以二进制写"wb",以二进制读"rb"…

文件的关闭函数fclose

参数是:fclose(打开文件时接收的指针)

这两个函数的使用方法如下:

#include "stdio.h"
#include "string.h"
#include "errno.h"
int main()
{
FILE* pf=fopen("text.txt","r")
//注意:text.txt是当前路径下的指定文件。
//若要打开其他地方的文件(如桌面),则要绝对路径
//例如:
//FILE* pf = fopen("C:\\Users\\cc\\Desktop\\test.txt", "r");
//读文件时要确保文件的存在
if(pf==NULL)
{
printf("%s\n",strerror(errno));
return 1;
}
//写文件
.....
//关闭文件
fclose(pf);
pf=NULL;
return 0;
}

运行后的结果是:

注意:我们输入的文件打开模式要用双引号"w”,“r”…,并且使用时要仔细检查输入的模式是否是我们需要的,因为使用错误时不会报警告,编译结果仍如上图所示。

三.文件的顺序读写

以下介绍的6个函数的头文件均为"stdio.h"

3.1 逐个字符的读写

这里需要使用两个函数:

字符输入函数:fgetc —>按字符读文件–>如果读取错误,则返回EOF

原型:int fgetc(FILE* strem)

参数:从文件指针指向的哪个文件读取

字符输出函数:fputc —>按字符写文件

原型:int fputc(int ch,FILE* strem)

参数:要写入的字符,要写入哪个文件

由于字符的本质是ASCII码,所以均是int类型。

以下是(写入)输出函数的用法:

#include "stdio.h"
#include "string .h"
#include "errno.h"
int main()
{
//打开文件
FILE* pf=fopen("test.txt","w");
//判断是否有该文件
if(pf==NULL)
{
printf("%s\n",strerror(errno));
return 1;
}
//写入文件
int i=0;
for(i='a';i<='z';i++)
{
fputc(i,pf);
}
//关闭文件
fclose(pf);
pf=NULL;
return 0;
}

运行结果为:

如图所示,在当前路径下的test.txt文件中已经写入了a~z字符。

以下是(读取)输入函数的用法:

#include "stdio.h"
#include "string.h"
#include "errno.h"
int main()
{
//打开文件
FILE* pf=fopen("test,txt","r")
//判断是否有该文件
if(pf==NULL)
{
printf("%s\n",strerror(errno));
return 1;
}
//读取文件
int ch=0;
while((ch=fgetc(pf))!=EOF)
{
printf("%c ",ch);
}
//关闭文件
fclose(pf);
pf=NULL;
return 0;
}

运行结果如图:

如图所示,在当前路径下的test.txt文件中读取了a~z字符。

3.2 逐行字符串的读写

这里需要使用两个函数:

文本行输入函数:fgets —>按行读文件–>读取失败,返回空指针

原型:int fgets(char* str,int num,FILE* strem)

参数:读取的数据放哪里,读取的个数,在哪个文件里读取

文本行输出函数:fputs —>按行写文件

原型:int fputs(const char* str,FILE* strem)

参数:需要写入的字符串,向指向的哪个文件里写

以下是(写入)输出函数的用法:

#include "stdio.h"
#include "string.h"
#include "errno.h"
int main()
{
//打开文件
FILE* pf=fopen("test.txt","w");
//检查
if(pf==NULL)
{
printf("%s\n",strerror(errno));
return 1;
}
//写一行数据
fputs("hello word",pf)
//关闭文件
fclose(pf);
pf=NULL;
return 0;
}

运行的结果为:

如图所示,在当前路径下的test.txt文件中已经写入了字符串。

以下是(读取)输入函数的用法:

int main()
{
  //打开文件
  FILE* pf = fopen("test.txt", "r");
  if (pf == NULL)
  {
    //printf("%s\n", strerror(errno));
    perror("fopen"); //也是打印错误信息的函数,上条语句的简化
    return 1;
  }
  //读一行数据
  char arr[20];//存放读取的数据
  fgets(arr,20, pf);
  printf("%s\n", arr);
  //关闭文件
  fclose(pf);
  pf = NULL;
  return 0;
}

运行的结果为:

如图所示,在当前路径下的test.txt文件中读取了字符串。

3.3 格式化读写函数

格式化输出函数,fprintf(FILE* strem, const char* str…)

格式化输入函数,fscanf(FILE* strem, const char* str…)

把结构体数据读写入文件中

以下是(写入)输出函数的用法:

#include "stdio.h"
#include "string.h"
#include "errno.h"
struct S 
{
char name[20];
int age;
};
int main()
{
struct S s;
//打开文件
FILE* pf=fopen("test.txt","w");
//检查
if(pf==NULL)
{
printf("%s\n",strerror(errno));
return 1;
}
//写入数据
fprintf(pf,"%s %d",s.name,s.age);//类比printf函数,只是多了一个写入的文件指针
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}

运行的结果为:

如图所示,在当前路径下的test.txt文件中写入了结构体里的数据。

以下是(读取)输入函数的用法:

#include "stdio.h"
#include "string.h"
#include "errno.h"
struct S
{
  char name[20];
  int age;
};
int main()
{
  struct S s = { 0 };
  //打开文件
  FILE* pf = fopen("test.txt", "r");
  if (pf == NULL)
  {
    //printf("%s\n", strerror(errno));
    perror("fopen"); //也是打印错误信息的函数,上条语句的简化
    return 1;
  }
  //
  fscanf(pf, "%s %d", s.name, &(s.age));//类比scanf函数,只是多了一个从哪里读的文件指针
  printf("%s %d", s.name, s.age);
  //关闭文件
  fclose(pf);
  pf = NULL;
  return 0;
}

运行的结果为:

如图所示,在当前路径下的test.txt文件中读取了结构体里的数据。

四.文件的随机读写

以下3个函数引用的头文件均为"stdio.h"

原型:int fseek(FILE* strem,long int offset,int origin)

根据文件指针的位置和便偏移量来定位文件指针

参数:所指向的文件,偏移量,当前文件指针的位置

位置有三个:

SEEK_SET–>该文件的起始位置

SEEK_CUR–>该文件指针当前指向的位置

SEEK-END–>该文件的末尾位置

ftell(FILE* strem)

返回文件指针相当于起始位置的偏移量

参数:所指向的文件

rewind(FILE* strem)

让文件指针的位置回到文件的起始位置

参数:所指向的文件

以下是使用方法:

#include "stdio.h"
#include "string.h"
#include "errno.h"
//读字符
int main()
{
//打开文件
FILE* pf("test.txt","r");
//检查
if(pf==NULL)
{
printf("%s\n",strerror(errno));
return 1;
}
//读文件,此时文件test.txt中有abcdef
fseek(pf,2,SEEK_SET)//定位文件指针,从起始位置(在a的前面)向右偏移两个指针
int ch=fgetc(pf);
printf("%c\n",ch);//c
printf("%d\n",ftell(pf));//3,相当于起始位置偏移三个量
//fseek(pf,2,SEEK_CUR);//当前位置指向c处,从当前位置(此时在d的前面)偏移两个指针
fseek(pf,-1,SEEK_END);//或是从末尾位置(f的后面)向左偏移一个指针,也可得到f
 ch=fgetc(pf);
printf("%c\n",ch);//f
printf("%d\n", ftell(pf));//6
//把文件指针还原到初始位置
rewind(pf);
ch=fgetc(pf);
printf("%c\n",ch);//a
printf("%d\n",ftell(pf));//1
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}

运行的结果为:


目录
相关文章
|
6天前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
22 6
|
19天前
|
存储 算法 程序员
C语言:库函数
C语言的库函数是预定义的函数,用于执行常见的编程任务,如输入输出、字符串处理、数学运算等。使用库函数可以简化编程工作,提高开发效率。C标准库提供了丰富的函数,满足各种需求。
|
25天前
|
机器学习/深度学习 C语言
【c语言】一篇文章搞懂函数递归
本文详细介绍了函数递归的概念、思想及其限制条件,并通过求阶乘、打印整数每一位和求斐波那契数等实例,展示了递归的应用。递归的核心在于将大问题分解为小问题,但需注意递归可能导致效率低下和栈溢出的问题。文章最后总结了递归的优缺点,提醒读者在实际编程中合理使用递归。
53 7
|
25天前
|
存储 编译器 程序员
【c语言】函数
本文介绍了C语言中函数的基本概念,包括库函数和自定义函数的定义、使用及示例。库函数如`printf`和`scanf`,通过包含相应的头文件即可使用。自定义函数需指定返回类型、函数名、形式参数等。文中还探讨了函数的调用、形参与实参的区别、return语句的用法、函数嵌套调用、链式访问以及static关键字对变量和函数的影响,强调了static如何改变变量的生命周期和作用域,以及函数的可见性。
29 4
|
25天前
|
存储 编译器 C语言
如何在 C 语言中判断文件缓冲区是否需要刷新?
在C语言中,可以通过检查文件流的内部状态或使用`fflush`函数尝试刷新缓冲区来判断文件缓冲区是否需要刷新。通常,当缓冲区满、遇到换行符或显式调用`fflush`时,缓冲区会自动刷新。
|
22天前
|
存储 C语言
【c语言】字符串函数和内存函数
本文介绍了C语言中常用的字符串函数和内存函数,包括`strlen`、`strcpy`、`strcat`、`strcmp`、`strstr`、`strncpy`、`strncat`、`strncmp`、`strtok`、`memcpy`、`memmove`和`memset`等函数的使用方法及模拟实现。文章详细讲解了每个函数的功能、参数、返回值,并提供了具体的代码示例,帮助读者更好地理解和掌握这些函数的应用。
19 0
|
22天前
|
C语言
【c语言】qsort函数及泛型冒泡排序的模拟实现
本文介绍了C语言中的`qsort`函数及其背后的回调函数概念。`qsort`函数用于对任意类型的数据进行排序,其核心在于通过函数指针调用用户自定义的比较函数。文章还详细讲解了如何实现一个泛型冒泡排序,包括比较函数、交换函数和排序函数的编写,并展示了完整的代码示例。最后,通过实际运行验证了排序的正确性,展示了泛型编程的优势。
18 0
|
6月前
|
存储 C语言
C 语言函数完全指南:创建、调用、参数传递、返回值解析
函数是一段代码块,只有在被调用时才会运行。 您可以将数据(称为参数)传递给函数。 函数用于执行某些操作,它们对于重用代码很重要:定义一次代码,并多次使用。
188 3
|
1月前
|
C语言
C语言函数返回值详解
本文详细解析了C语言中函数返回值的概念与应用。从函数的基本定义入手,深入探讨了不同类型返回值的作用及意义,并提供了实用的编程示例,帮助读者更好地理解和使用函数返回值。通过本文,你将掌握如何有效利用返回值优化代码结构与功能实现。
|
5月前
|
存储 C语言
C语言的函数返回值和指针
C|函数返回值(区分各类值)和指针(区分各类存储空间)的细节