工作中碰到一个问题,如何只修改文件中间的几个字节,而其他的内容不变。这个问题看似简单,但是很多人估计都不知道怎么做。我开始seek到文件的特定的位置,然后写文件,但是使用的文件打开模式不对,文件不是被清空,就是被截断,达不到效果。
在C语言中文件打开方式有这么几种:
另外还有2个选项,可以与上面的6个选项复合使用,一个是t表示以文本的方式打开文件(默认是t),一个是b表示以二进制的方式打开文件,t和b是互斥的不能同时使用。当与b组合时,有这么几种方式:wb、ab、rb、wb+、ab+、rb+,而a,w,r这几个选项是不能组合使用的,其中a,w都表示写文件,只不过一个在文件尾,一个在文件开始处,r表示读文件。我试过将a,w,r几个两两组合使用,发现下面的现象:
可以看出来当a,w,r在一起组合使用的时候,其后面的选项实际上好像是被忽略了
所以解决文章开头提出来的问题,应该使用 rb+ 的方式打开文件,这种方式打开的文件,可读,可写,打开之后写指针在文件开始处,可以任意seek,而seek之后写的内容会覆盖被写的内容,其他没有写到的内容不会有改变。
//程序测试结果在ubuntu linux下运行获得 #include <stdio.h> #include <string.h> int main() { //文件原始数据 //00 01 02 03 04 05 06 07 08 09 //下面每一个fopen前面注释中的数据是以该方式打开文件,写文件之后文件的内容 //00 00 00 00 CC DD //FILE * file = fopen("./test.data","wb+"); //00 00 00 00 CC DD //FILE * file = fopen("./test.data","wb"); //00 01 02 03 CC DD 06 07 08 09 FILE * file = fopen("./test.data","rb+"); //这种是正确的做法 //00 01 02 03 04 05 06 07 08 09 //FILE * file = fopen("./test.data","rb"); //00 01 02 03 04 05 06 07 08 09 CC DD //FILE * file = fopen("./test.data","ab"); //00 01 02 03 04 05 06 07 08 09 CC DD //FILE * file = fopen("./test.data","ab+"); //00 00 00 00 CC DD //FILE * file = fopen("./test.data","wr"); //00 01 02 03 04 05 06 07 08 09 //FILE * file = fopen("./test.data","rw"); //00 01 02 03 04 05 06 07 08 09 CC DD //FILE * file = fopen("./test.data","aw"); //00 00 00 00 CC DD //FILE * file = fopen("./test.data","wa"); //00 01 02 03 04 05 06 07 08 09 CC DD //FILE * file = fopen("./test.data","ar"); //00 01 02 03 04 05 06 07 08 09 //FILE * file = fopen("./test.data","ra"); if(file!=NULL) { char buffer[]={0xCC,0xDD}; fseek(file,4,SEEK_SET); fwrite(buffer,1,sizeof(buffer),file); fclose(file); } return 0; }