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

相关文章

C/C++错误信息处理的常见方法及函数

《C/C++错误信息处理的常见方法及函数》C/C++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(

Kotlin 作用域函数apply、let、run、with、also使用指南

《Kotlin作用域函数apply、let、run、with、also使用指南》在Kotlin开发中,作用域函数(ScopeFunctions)是一组能让代码更简洁、更函数式的高阶函数,本文将... 目录一、引言:为什么需要作用域函数?二、作用域函China编程数详解1. apply:对象配置的 “流式构建器”最

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

C语言中的数据类型强制转换

《C语言中的数据类型强制转换》:本文主要介绍C语言中的数据类型强制转换方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C语言数据类型强制转换自动转换强制转换类型总结C语言数据类型强制转换强制类型转换:是通过类型转换运算来实现的,主要的数据类型转换分为自动转换

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件

C语言实现两个变量值交换的三种方式

《C语言实现两个变量值交换的三种方式》两个变量值的交换是编程中最常见的问题之一,以下将介绍三种变量的交换方式,其中第一种方式是最常用也是最实用的,后两种方式一般只在特殊限制下使用,需要的朋友可以参考下... 目录1.使用临时变量(推荐)2.相加和相减的方式(值较大时可能丢失数据)3.按位异或运算1.使用临时

使用C语言实现交换整数的奇数位和偶数位

《使用C语言实现交换整数的奇数位和偶数位》在C语言中,要交换一个整数的二进制位中的奇数位和偶数位,重点需要理解位操作,当我们谈论二进制位的奇数位和偶数位时,我们是指从右到左数的位置,本文给大家介绍了使... 目录一、问题描述二、解决思路三、函数实现四、宏实现五、总结一、问题描述使用C语言代码实现:将一个整

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++中函数模板与类模板的简单使用及区别介绍

《C++中函数模板与类模板的简单使用及区别介绍》这篇文章介绍了C++中的模板机制,包括函数模板和类模板的概念、语法和实际应用,函数模板通过类型参数实现泛型操作,而类模板允许创建可处理多种数据类型的类,... 目录一、函数模板定义语法真实示例二、类模板三、关键区别四、注意事项 ‌在C++中,模板是实现泛型编程

kotlin的函数forEach示例详解

《kotlin的函数forEach示例详解》在Kotlin中,forEach是一个高阶函数,用于遍历集合中的每个元素并对其执行指定的操作,它的核心特点是简洁、函数式,适用于需要遍历集合且无需返回值的场... 目录一、基本用法1️⃣ 遍历集合2️⃣ 遍历数组3️⃣ 遍历 Map二、与 for 循环的区别三、高