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

运行的结果为:


目录
相关文章
|
9天前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
33 6
|
26天前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
31 6
|
1月前
|
存储 算法 程序员
C语言:库函数
C语言的库函数是预定义的函数,用于执行常见的编程任务,如输入输出、字符串处理、数学运算等。使用库函数可以简化编程工作,提高开发效率。C标准库提供了丰富的函数,满足各种需求。
|
2月前
|
机器学习/深度学习 C语言
【c语言】一篇文章搞懂函数递归
本文详细介绍了函数递归的概念、思想及其限制条件,并通过求阶乘、打印整数每一位和求斐波那契数等实例,展示了递归的应用。递归的核心在于将大问题分解为小问题,但需注意递归可能导致效率低下和栈溢出的问题。文章最后总结了递归的优缺点,提醒读者在实际编程中合理使用递归。
68 7
|
2月前
|
存储 编译器 程序员
【c语言】函数
本文介绍了C语言中函数的基本概念,包括库函数和自定义函数的定义、使用及示例。库函数如`printf`和`scanf`,通过包含相应的头文件即可使用。自定义函数需指定返回类型、函数名、形式参数等。文中还探讨了函数的调用、形参与实参的区别、return语句的用法、函数嵌套调用、链式访问以及static关键字对变量和函数的影响,强调了static如何改变变量的生命周期和作用域,以及函数的可见性。
33 4
|
2月前
|
存储 C语言
【c语言】字符串函数和内存函数
本文介绍了C语言中常用的字符串函数和内存函数,包括`strlen`、`strcpy`、`strcat`、`strcmp`、`strstr`、`strncpy`、`strncat`、`strncmp`、`strtok`、`memcpy`、`memmove`和`memset`等函数的使用方法及模拟实现。文章详细讲解了每个函数的功能、参数、返回值,并提供了具体的代码示例,帮助读者更好地理解和掌握这些函数的应用。
32 0
|
2月前
|
C语言
【c语言】qsort函数及泛型冒泡排序的模拟实现
本文介绍了C语言中的`qsort`函数及其背后的回调函数概念。`qsort`函数用于对任意类型的数据进行排序,其核心在于通过函数指针调用用户自定义的比较函数。文章还详细讲解了如何实现一个泛型冒泡排序,包括比较函数、交换函数和排序函数的编写,并展示了完整的代码示例。最后,通过实际运行验证了排序的正确性,展示了泛型编程的优势。
23 0
|
2月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
37 3
|
2月前
|
存储 缓存 C语言
【c语言】简单的算术操作符、输入输出函数
本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
44 10
|
2月前
|
存储 编译器 C语言
C语言函数的定义与函数的声明的区别
C语言中,函数的定义包含函数的实现,即具体执行的代码块;而函数的声明仅描述函数的名称、返回类型和参数列表,用于告知编译器函数的存在,但不包含实现细节。声明通常放在头文件中,定义则在源文件中。