c语言读一字节程序,C语言读写二进制文件

  • Post author:
  • Post category:其他


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

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

完整代码:

#include

#include

#include

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内容给更改了。