【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;
}

运行的结果为:


目录
相关文章
|
1天前
|
编译器 C语言
C语言编译详解:GCC分步编译与一次编译多个文件
C语言编译详解:GCC分步编译与一次编译多个文件
10 2
|
1天前
|
存储 C语言
C语言文件读写详解
C语言文件读写详解
10 1
|
2天前
|
机器学习/深度学习 C语言
详细解读C语言math.h中常用函数
详细解读C语言math.h中常用函数
|
2天前
|
C语言
C语言刷题(函数)
C语言刷题(函数)
|
1天前
|
C语言
C语言中的函数指针、指针函数与函数回调
C语言中的函数指针、指针函数与函数回调
6 0
|
1天前
|
存储 C语言
C语言中的变量与函数详解
C语言中的变量与函数详解
2 0
|
1天前
|
存储 C语言
C语言中的printf函数详解
C语言中的printf函数详解
8 0
|
7月前
|
C语言
【C语言】用函数递归的方法解决汉诺塔问题
【C语言】用函数递归的方法解决汉诺塔问题
37 0
|
算法 程序员 C语言
C语言基础(有关三个数比较大小、冒泡排序、最大公约数、和有关某个数x的绝对值的n次方除于n的阶乘问题的函数求解法;和阶乘函数递归方法;和数组作函数参数的
C语言基础(有关三个数比较大小、冒泡排序、最大公约数、和有关某个数x的绝对值的n次方除于n的阶乘问题的函数求解法;和阶乘函数递归方法;和数组作函数参数的
|
C语言
字符串逆序(C语言版 函数,递归方法)
字符串逆序(C语言版 函数,递归方法)
104 0
字符串逆序(C语言版 函数,递归方法)