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

相关文章

Python函数返回多个值的多种方法小结

《Python函数返回多个值的多种方法小结》在Python中,函数通常用于封装一段代码,使其可以重复调用,有时,我们希望一个函数能够返回多个值,Python提供了几种不同的方法来实现这一点,需要的朋友... 目录一、使用元组(Tuple):二、使用列表(list)三、使用字典(Dictionary)四、 使

Mysql中的用户管理实践

《Mysql中的用户管理实践》:本文主要介绍Mysql中的用户管理实践,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录13. 用户管理13.1 用户 13.1.1 用户信息 13.1.2 创建用户 13.1.3 删除用户 13.1.4 修改用户

Go语言中使用JWT进行身份验证的几种方式

《Go语言中使用JWT进行身份验证的几种方式》本文主要介绍了Go语言中使用JWT进行身份验证的几种方式,包括dgrijalva/jwt-go、golang-jwt/jwt、lestrrat-go/jw... 目录简介1. github.com/dgrijalva/jwt-go安装:使用示例:解释:2. gi

Go 语言中的 Struct Tag 的用法详解

《Go语言中的StructTag的用法详解》在Go语言中,结构体字段标签(StructTag)是一种用于给字段添加元信息(metadata)的机制,常用于序列化(如JSON、XML)、ORM映... 目录一、结构体标签的基本语法二、json:"token"的具体含义三、常见的标签格式变体四、使用示例五、使用

PyTorch中cdist和sum函数使用示例详解

《PyTorch中cdist和sum函数使用示例详解》torch.cdist是PyTorch中用于计算**两个张量之间的成对距离(pairwisedistance)**的函数,常用于点云处理、图神经网... 目录基本语法输出示例1. 简单的 2D 欧几里得距离2. 批量形式(3D Tensor)3. 使用不

linux服务之NIS账户管理服务方式

《linux服务之NIS账户管理服务方式》:本文主要介绍linux服务之NIS账户管理服务方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、所需要的软件二、服务器配置1、安装 NIS 服务2、设定 NIS 的域名 (NIS domain name)3、修改主

MySQL 字符串截取函数及用法详解

《MySQL字符串截取函数及用法详解》在MySQL中,字符串截取是常见的操作,主要用于从字符串中提取特定部分,MySQL提供了多种函数来实现这一功能,包括LEFT()、RIGHT()、SUBST... 目录mysql 字符串截取函数详解RIGHT(str, length):从右侧截取指定长度的字符SUBST

Go语言使用slices包轻松实现排序功能

《Go语言使用slices包轻松实现排序功能》在Go语言开发中,对数据进行排序是常见的需求,Go1.18版本引入的slices包提供了简洁高效的排序解决方案,支持内置类型和用户自定义类型的排序操作,本... 目录一、内置类型排序:字符串与整数的应用1. 字符串切片排序2. 整数切片排序二、检查切片排序状态:

基于Go语言实现Base62编码的三种方式以及对比分析

《基于Go语言实现Base62编码的三种方式以及对比分析》Base62编码是一种在字符编码中使用62个字符的编码方式,在计算机科学中,,Go语言是一种静态类型、编译型语言,它由Google开发并开源,... 目录一、标准库现状与解决方案1. 标准库对比表2. 解决方案完整实现代码(含边界处理)二、关键实现细

如何合理管控Java语言的异常

《如何合理管控Java语言的异常》:本文主要介绍如何合理管控Java语言的异常问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、介绍2、Thorwable类3、Error4、Exception类4.1、检查异常4.2、运行时异常5、处理方式5.1. 捕获异常