C语言读写二进制文件

可以这么说,除了文本文件以外的所有文件都是二进制文件。二进制文件相对于文本文件更容易修改。因为文本文件的修改,需要修改以后写入内存,然后再清空原文件,再从内存中读取出修改以后的内容到本文件中。二进制文件修改只需要找到特定的字符,然后对它进行修改。

对二进制文件进行操作主要用到fopen,fread,fwrite,fseek等函数。

完整代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

struct stu
{
    int id;
    char name[10];
    int age;
};    //定义一个学生结构体

struct stu s1;

void read_file(char *filename)
{
    FILE *fd = fopen(filename,"rb"); 
    if(fd == NULL)
    {
        perror("open failed!");
        exit(1);        //出错、退出
    }

    fread(&s1,sizeof(s1),1,fd);
    printf("%d %d %s\n",s1.id,s1.age,s1.name);

    fclose(fd);
}

//写入文件
void write_file(char *filename)
{
    FILE *fd = fopen(filename,"wb");
    s1.id=77;
    s1.age=17;
    strcpy(s1.name,"yang");

    if(fd == NULL)
    {
        perror("open failed!");
        exit(1);
    }

    fwrite(&s1,sizeof(s1),1,fd);
    fclose(fd);

}

//修改文件
void modify(char *filename)
{
    FILE *fd = fopen(filename,"rb+");
    if(fd == NULL)
    {
        perror("open failed!");
        exit(1);
    }


    while(1)
    {

        int res = fread(&s1,sizeof(s1),1,fd);
        if(res <= 0)
        {
            break;
        }

        if(s1.id == 77)  //通过查找id=77的位置来修改该结构体成员
        {
            s1.age=22;
            strcpy(s1.name,"yangni");
            fseek(fd,-sizeof(s1),SEEK_CUR); //设置偏移量
            fwrite(&s1,sizeof(s1),1,fd);
            break;
        }
    }


}


int main()
{
    write_file("test.txt");
    read_file("test.txt");
    modify("test.txt");

    printf("%d %d %s\n",s1.id,s1.age,s1.name);

    read_file("test.txt");

    return 0;
}

主要函数:

1、fopen()

FILE * fopen(const char * path, const char * mode);

FILE *fd = fopen(filename,"rb"); 
    if(fd == NULL)
    {

        perror("open failed!");
        exit(1);        //出错、退出
    }
  • path字符串包含欲打开的文件路径及文件名
  • 参数 mode 表示文件打开方式

参数mode:

字符串说明
r以只读方式打开,且文件必须存在
r+以读/写方式打开文件,该文件必须存在
rb+以读/写方式打开一个二进制文件,只允许读/写数据
rt+以读/写方式打开一个文本文件,允许读和写
w打开只写文件,若文件存在则长度清为 0,即该文件内容消失,若不存在则创建该文件
w+打开可读/写文件,若文件存在则文件长度清为零,即该文件内容会消失。若文件不存在则建立该文件。
a附加的方式打开只写文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾,即文件原先的内容会被保留(EOF 符保留)
a+以附加方式打开可读/写的文件。若文件不存在,则会建立该文件,如果文件存在,则写入的数据会被加到文件尾后,即文件原先的内容会被保留(原来的 EOF 符不保留)
wb以只写方式打开或新建一个二进制文件,只允许写数据
wb+以读/写方式打开或建立一个二进制文件,允许读和写
ab+以读/写方式打开一个文本文件,允许读或在文本末追加数据

2、fread()

size_t fread ( void *buffer, size_t size, size_t count, FILE *stream) ;

返回值:返回实际写入的数据块数目

从文件流中读数据,最多读取count个项,每个项size个字节,如果调用成功返回实际读取到的项个数(小于或等于count),如果不成功或读到文件末尾返回0

  • buffer 用于接收数据的内存地址

  • size 要读的每个数据项的字节数,单位是字节

  • count 要读count个数据项,每个数据项size个字节.

  • stream 输入流


3、fwrite()

size_t fwrite(const void* buffer, size_t size, size_t count, FILE* stream);

注意:这个函数以二进制形式对文件进行操作,不局限于文本文件

返回值:返回实际写入的数据块数目

(1)buffer:是一个指针,对fwrite来说,是要获取数据的地址;

(2)size:要写入内容的单字节数;

(3)count:要进行写入size字节的数据项的个数;

(4)stream:目标文件指针;

(5)返回实际写入的数据项个数count。


4、fseek()

int fseek(FILE *stream, long offset, int fromwhere);

函数设置文件指针stream的位置。如果执行成功,stream将指向以fromwhere(偏移起始位置:文件头0(SEEK_SET),当前位置1(SEEK_CUR),文件尾2(SEEK_END))为基准,偏移offset(指针偏移量)个字节的位置。如果执行失败(比如offset超过文件自身大小),则不改变stream指向的位置。

(1)stream:文件指针

(2) offset:为偏移量,正数表示正向偏移,负数表示负向偏移

(3) fromwhere:从文件的哪里开始偏移,可能取值为:SEEK_SET,SEEK_CUR、 SEEK_END (对应0 、 1 、 2)

fseek(fp,100L,0); 把stream指针移动到离文件开头100字节处;

fseek(fp,100L,1); 把stream指针移动到离文件当前位置100字节处;

fseek(fp,-100L,2); 把stream指针退回到离文件结尾100字节处。

关于fseek上文代码分析:

if(s1.id == 77)  //通过查找id=77的位置来修改该结构体成员
{
    s1.age=22;
    strcpy(s1.name,"yangni");
    fseek(fd,-sizeof(s1),SEEK_CUR); //设置偏移量
    fwrite(&s1,sizeof(s1),1,fd);
    break;
}

通过查找s1.id获得s1对象,此时光标锁定到了该对象数据末尾,因此修改完数据后。可设置向左偏移该结构体长度,将光标偏移到该结构体之前,再写入数据就可以覆盖之前的s1,就把s1内容给更改了。

相关推荐
©️2020 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页