C语言中的各种文件读写方法小结
作者:低调小一 发布时间:2023-06-13 02:44:40
前言
找工作的时候,曾经用C语言练习过一段时间的算法题目,也在几个还算出名的OJ平台有过还算靠谱的排名。之前以为C语言只限于练习一下算法,但是工作中的一个问题解决让我意识到C语言的用处还是非常广泛的。下面介绍一下,如果用C语言来操作文件保存一个字符串,和读取一个字符串。算法中往往都是printf来打印出结果,但是真实工作中往往通过文件来进行一些持久化的存储工作。
C-File I/O
文件的I/O操作是每一门语言的重点,因此这里我先来介绍一下如何用C语言去进行文件的I/O操作。
文件和流
就C语言程序而言,所有的I/O操作只是简单地从程序移进或移出字节的事情。因此,这种字节流便被称为流(stream)。程序只需要关心创建正确的输出字节数据,以及正确地解释从输入读取的字节数据。特定I/O设备的细节对程序员是隐藏的。绝大多数流是完全缓冲的(fully buffered),这意味着“读取”和“写入”实际上是从一块被称为缓冲区(buffer)的内存区域来回复制数据。从内存中来回复制数据是非常快速的。用于输出流的缓冲区只有当它写满时才会被刷新(flush,物理写入)到设备或文件中。一次性把写满的缓冲区写入和逐片把程序产生的输出分别写入相比效率更高。输入缓冲区也是类似的原理。
流分为两种类型,分别是文本流和二进制流。
打开流和关闭流
fopen函数打开一个特定的文件,并把一个流和这个文件相关联。它的原型如下所示:
[cpp] view plaincopyprint?在CODE上查看代码片派生到我的代码片
FILE* open(const char* name, const char* mode);
name参数是你希望打开的文件或设备的名字。mode参数标识流用于只读、只写还是既读又写,以及它是文本流还是二进制流。下面表格里列出了一些常用的模式:
如果fopen函数执行成功,它将返回一个指向FILE结构的指针,该结构代表这个新创建的流。如果函数执行失败,它将返回一个NULL指针,error会提示问题的性质。
流是用函数fclose关闭的,它的原型如下:
[cpp] view plaincopyprint?在CODE上查看代码片派生到我的代码片
int fclose(FILE *f);
对于输出流,fclose函数在文件关闭之前刷新缓冲区。如果它执行成功,fclose返回零值,否则返回EOF。
由于fopen和fclose打开和关闭的都是FILE结构体指针,而在stdio.h头文件中,包含了对文件结构体FILE的描述。这里介绍一下FILE结构体定义:
struct _iobuf {
char *_ptr; // 下一个要被读取的字符的地址
int _cnr; // 剩余的字符
char *base; // 缓冲区基地址
int _flag; // 读写文件标志位
int _file; // 文件号
int _charbuf; // 检查缓冲区的状况
int _bufsiz; // 文件的大小
char *_tmpfname; // 临时文件名
};
typedef struct _iobuf FILE;
字符I/O
当一个流被打开之后,它可以用于输入和输出。它最简单的形式是字符I/O。字符输入是由getchar函数家族执行的,它们的原型如下所示:
int fgetc(FILE *stream);
int getc(FILE *stream);
int getchar(void);
需要操作的流作为参数传递给getc和fgetc,但是getchar始终是从标准输入读取。每个函数从流中读取下一个字符,并把它作为函数的返回值返回。如果流中不存在更多的字符,函数就返回常量值EOF(-1)。
为了把单个字符写入到流中,可以使用putchar函数家族。它的原型如下:
[cpp] view plaincopyprint?在CODE上查看代码片派生到我的代码片
int fputc(int character, FILE* stream);
int putc(int character, FILE* stream);
int putchar(int character);
行I/O
行I/O其实可以用两种方式执行——未格式化的或者格式化的。这两种形式都用于操纵字符串。区别在于未格式化的I/O只是通过fgets和fputs简单读取或写入字符串,而格式化的I/O则执行数字和其他变量的内部或外部表示形式之间的转换。由于日常工作中操作的一般都是格式化I/O,因此这里不讲fgets和fputs这种非格式化I/O操作了。(当然,还有一个重要的原因,fgets无法判断缓冲区长度,容易导致溢出等情况)
scanf家族
scanf函数家族的原型如下所示。每个原型中的省略号表示一个可变长度的指针列表。从输入转换而来的值逐个存储到这些指针参数所指向的内存位置。
int fscanf(FILE* stream, const char* format, ...);
int scanf(const char* format, ...);
int sscanf(const char* string, const char* format, ...);
这些函数都从输入源读取字符并根据format字符串给出的格式化代码对它们进行转换。当格式化字符串到达末尾或者读取的输入不再匹配格式字符串所指定的类型时,输入就停止。在任何一种情况下,被转换的输入值的数目作为函数的返回值返回。如果在任何输入值被转换之前文件就已经到达尾部,函数就返回常量值EOF。
printf家族
printf函数家族用于创建格式化的输出。它们的函数原型如下:
int fprintf(FILE *stream, const char* format, ...);
int printf(const char* format, ...);
int sprintf(char* buffer, const char* format, ...);
二进制I/O
把数据写到文件里效率最高的方法是用二进制形式写入,而且Android系统里也有很有用二进制文件通过位来存储数据的应用场景。介绍一下操纵二进制I/O的函数原型。
fread函数用于读取二进制数据,fwrite函数用于写入二进制数据。它们的原型如下所示:
[cpp] view plaincopyprint?在CODE上查看代码片派生到我的代码片
size_t fread(void* buffer, size_t size, size_t count, FILE* stream);
size_t fwrite(void* buffer, size_t size, size_t count, FILE* stream);
buffer是一个指向用于保存数据的内存位置的指针,size是缓冲区中每个元素的字节数,count是读取或写入的元素数,stream是数据读取或写入的流。
刷新和定位函数
在处理流时,另外还有一些函数也较为有用。首先,是fflush,它迫使一个输出流的缓冲区内的数据进行物理写入,不管它是不是已经写满。它的原型如下所示:
int fflush(FILE* stream);
当我们需要立即把输出缓冲区的数据进行物理写入时,应该使用这个函数。
在正常的情况下,数据以线性的方式写入,这意味着后面写入的数据在文件中的位置是在以前所有写入数据的后面。C同时支持随机访问I/O,也就是以任意顺序访问文件的不同位置。随机访问是通过在读取或写入前先定位到文件中需要的位置来实现的。一般使用fseek函数来实现,函数原型如下:
int fseek(FILE* stream, long offset, int from);
fseek函数允许你在一个流中定位。这个操作将改变下一个读取或写入的位置。它的第一个参数是需要改变的流,它的第二个和第三个参数标识文件中需要定位的位置。下表描述了fseek参数的使用方法。


猜你喜欢
- SpringMVC服务器验证一种是有两种方式,一种是基于Validator接口,一种是使用Annotaion JSR-303标准的验证,下面
- 遇到的问题解决方法win7:insert键切回win10:fn+insert切回Mac:fn+i 切回来源:https://blog.csd
- 本篇内容通过操作软键盘的函数着手详细分析了隐藏或者显示软键盘的实现方法,并且对其中重要的代码做了详细分析。一、开篇如果有需要用到输入的地方,
- 本文实例为大家分享了Java实现聊天机器人完善版的具体代码,供大家参考,具体内容如下Client代码:package GUISocket.c
- 在使用Mybatis时,有的时候我们可以不用定义resultMap,而是直接在<select>语句上指定resultType。这
- 大致思路:注解实现方式:就是用 反射机制. 获取指定的包下使用了注解的类,存储在一个map容器, 然后获取map容器下类的属性, 利用反射给
- 我们在写linux的服务的时候,经常会用到linux的多线程技术以提高程序性能 多线程的一些小知识:一个应用程序可以启动若干个线程
- 本文实例为大家分享了C#实现学生成绩管理系统的具体代码,供大家参考,具体内容如下C#作业 用循环结构/数组实现进入启动页面管理员身份登入用户
- [LeetCode] 159. Longest Substring with At Most Two Distinct Characters
- 多态性1理解多态性:可以理解为一个事物的多种形态。2何为多态性:对象的多态性:父类的引用指向子类的对象(或子类的对象赋给父类的引用)3多态的
- JNDI的理解JNDI是 Java 命名与文件夹接口(Java Naming and Directory Interface),在J2EE规
- 本文设计一个简单的班级管理系统,满足如下要求:1、设计学生类Student,包含学号(String型)、姓名(String型)、
- 本文实例分析了Android TabHost选项卡标签图标始终不出现的解决方法。分享给大家供大家参考,具体如下:在学习Android Tab
- public class MyGestureLintener extends SimpleOnGestureListener {privat
- 前言app启动后的白屏问题,默认都是在splash页面加主题配置,主题配置一个背景来达到用户点击app图标就立马启动app的假象,大多情况下
- 前言嗯。最近工程上遇到一个byte数组转换为int的问题,解决过程中遇到了几个坑,经过各种查资料终于还是解决了。撒花。Java的位运算以及b
- 写在前面在Java8之前的版本中,接口中只能声明常量和抽象方法,接口的实现类中必须实现接口中所有的抽象方法。而在Java8中,接口中可以声明
- 一、概述UDP和TCP是网络通讯常用的两个传输协议,C#一般可以通过Socket来实现UDP和TCP通讯,由于.NET框架通过UdpClie
- 搭建测试的数据库和表数据的SQL/* Navicat Premium Data TransferSource Server &
- 前言之前在SpringBoot项目中简单使用定时任务,不过由于要借助cron表达式且都提前定义好放在配置文件里,不能在项目运行中动态修改任务