C语言文件管理之二 fwrite函数和fread函数

2023-11-03 20:08

本文主要是介绍C语言文件管理之二 fwrite函数和fread函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

第一篇文章已经介绍了fopen函数。而且也使用了fwrite和fread函数,但并没有介绍这两个函数。这篇详细解释这两个函数的使用。
顾名思义,fwrite函数是用来往文件写内容的,fread函数是可以从文件读取内容的。
首先要明白一点,这个两个函数是可以读写二进制内容,但是作为例子,我们是看不懂二进制内容的,所以还是以读写文本为例子。还有读写是存在编码问题的,为了更简洁的介绍这两个函数,我们不使用中文字符串(会有乱码),(编码问题单独写一篇讨论)。

先看下fwrite函数的声明。

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

fwrite有四个参数,但源码取的名字非常的难理解。需要解释一下

  • 第1个参数:const void *类型的指针,其实是要写的内容,可能是任意类型,不一定是字符串。而且是可以写int类型的,但是几乎是不可以用的,因为会把int当成4个char来写入。 而且没有用满的地方写入nul,这是不能接受的。
  • 第2个参数:size 表示的是一个写入单位的大小,比如类型可能int,那么size就是4。如果是一个struct,那么就是这个struct的大小。
  • 第3个参数:n表示数量 计算方式是 总长度/单位大小。在GNU man手册中有个这样的宏定义#define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0])), ARRAY_SIZE(arr)表示的就是这里的n。
  • 第4个参数:stream 这个好理解,需要操作的文件流。
    举个例子就明白了。这个例子的目的就是往一个文件里面写入字符串。
int main() {FILE *file = fopen("test.txt", "w+");if (file == NULL) {perror("fopen");EXIT_FAILURE;}char buf[] = "abcd";size_t ret = fwrite(buf, sizeof(*buf) , sizeof(buf)/ sizeof(buf[0])-1, file);if (ret != sizeof(buf)) {perror("fwrite");EXIT_FAILURE;}return 0;
}

我们定义了一个char数组buf。内容为"abcd\0"。注意在C语言里面字符串是会自动在后面加上\0结束符号的,所以数组的实际大小是5。我们关系的fwrite函数传入的参数

size_t ret = fwrite(buf, sizeof(*buf) , sizeof(buf)/ sizeof(buf[0])-1, file);

这行代码等价于:

size_t ret = fwrite(buf, 1, 5-1, file);

第二个参数的传入的是char类型占用内存大小,也就是占用1个字节。
第三个参数表示数组总长度/单个长度,其实就是数组数量,因为字符串会自动加\0,所以我们需要手动将结果减1。这样就可以写入4个char ‘a’ ‘b’ ‘c’ ’d‘。如果不减1,结果就会变成下面这样

这显然不是我们想要的。
事实上,正是因为每次都要注意结尾的这个\0,并不是特别的方便。所以有了fprintf和fputs这些专门用来操作字符串的函数,用起来更加方便。关于这两个函数,会专门写一篇,这里为了这篇文章的简洁性,就不深入讨论了。
事实上,fwrite这个函数是比较“难用”的。因为他的功能更强,体现在他可以操作void *类型的数据,也就是什么数据都可以写。当然我们只能看懂char类型的数据,其它数据只有机器才能看懂,在我们看来就是乱码。
比如写入int类型,看下面的例子:

int main() {FILE *file = fopen("test.txt", "w+");if (file == NULL) {perror("fopen");EXIT_FAILURE;}int buf[]={66,67,68,69};size_t ret = fwrite(buf, sizeof(*buf) , sizeof(buf)/ sizeof(buf[0]), file);if (ret != sizeof(buf)) {perror("fwrite");EXIT_FAILURE;}return 0;
}

输出的结果是这样的:
在这里插入图片描述
仔细一看,还是能看懂部分内容的,也就是字符 bcde。也就是对应66,67,68,69。但是int是占用4个字节的,多出来的三个字节用NUL填充。所以写入int这种做法只能当例子举举,真要用,你也只能写成“66676869”这种字符串。因为这是人可以看懂的。
当然对于程序员来说,我们常见的需求可能就是写文本文件,而不是写二进制文件。所以fprintf和fputs函数可能用起来更方便。而不是fwrite函数。
说完了fwrite函数,fread函数就简单了,他们的参数是一样的。
看下面的例子:

int main() {FILE *file = fopen("test.txt", "w+");if (file == NULL) {perror("fopen");EXIT_FAILURE;}int buf[]={66,67,68,69};size_t ret = fwrite(buf, sizeof(*buf) , sizeof(buf)/ sizeof(buf[0]), file);if (ret != sizeof(buf)) {perror("fwrite");EXIT_FAILURE;}//新增代码fseek(file,0,SEEK_SET);char readbuf[3];fread(readbuf, sizeof(*buf), sizeof(buf)/sizeof(buf[0]),file);printf("%c\n",readbuf[0]);printf("%c\n",readbuf[1]);printf("%c\n",readbuf[2]);return 0;
}

这里我们用到了fseek函数,因为我们是先写入数据到文件,然后再读,在数据写入完成后,指针是指向文件末尾的,也就是末尾后面的内容是内存中未知内容。所以我们需要把指针定位到文件开头。 fseek(file,0,SEEK_SET);这行代码就是把指针定位到文件开头,知道是这个作用就行,fseek具体细节会专门写一篇文章。这样就可以通过fread函数读取文件的内容了。读到的结果保存在我们定义的buf变量里面。
结果如下:

a
b
c

这篇关于C语言文件管理之二 fwrite函数和fread函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/340570

相关文章

Go语言中make和new的区别及说明

《Go语言中make和new的区别及说明》:本文主要介绍Go语言中make和new的区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1 概述2 new 函数2.1 功能2.2 语法2.3 初始化案例3 make 函数3.1 功能3.2 语法3.3 初始化

MySQL 中的 CAST 函数详解及常见用法

《MySQL中的CAST函数详解及常见用法》CAST函数是MySQL中用于数据类型转换的重要函数,它允许你将一个值从一种数据类型转换为另一种数据类型,本文给大家介绍MySQL中的CAST... 目录mysql 中的 CAST 函数详解一、基本语法二、支持的数据类型三、常见用法示例1. 字符串转数字2. 数字

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

MySQL count()聚合函数详解

《MySQLcount()聚合函数详解》MySQL中的COUNT()函数,它是SQL中最常用的聚合函数之一,用于计算表中符合特定条件的行数,本文给大家介绍MySQLcount()聚合函数,感兴趣的朋... 目录核心功能语法形式重要特性与行为如何选择使用哪种形式?总结深入剖析一下 mysql 中的 COUNT

Go语言中nil判断的注意事项(最新推荐)

《Go语言中nil判断的注意事项(最新推荐)》本文给大家介绍Go语言中nil判断的注意事项,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1.接口变量的特殊行为2.nil的合法类型3.nil值的实用行为4.自定义类型与nil5.反射判断nil6.函数返回的

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

MySQL 中 ROW_NUMBER() 函数最佳实践

《MySQL中ROW_NUMBER()函数最佳实践》MySQL中ROW_NUMBER()函数,作为窗口函数为每行分配唯一连续序号,区别于RANK()和DENSE_RANK(),特别适合分页、去重... 目录mysql 中 ROW_NUMBER() 函数详解一、基础语法二、核心特点三、典型应用场景1. 数据分

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN

Python get()函数用法案例详解

《Pythonget()函数用法案例详解》在Python中,get()是字典(dict)类型的内置方法,用于安全地获取字典中指定键对应的值,它的核心作用是避免因访问不存在的键而引发KeyError错... 目录简介基本语法一、用法二、案例:安全访问未知键三、案例:配置参数默认值简介python是一种高级编