zoukankan      html  css  js  c++  java
  • 如何修改文件中间的几个字节

    工作中碰到一个问题,如何只修改文件中间的几个字节,而其他的内容不变。这个问题看似简单,但是很多人估计都不知道怎么做。我开始seek到文件的特定的位置,然后写文件,但是使用的文件打开模式不对,文件不是被清空,就是被截断,达不到效果。

    fopen的打开模式

    在C语言中文件打开方式有这么几种:

    • r   以只读方式打开文件,只能读不能写,往文件中写是没有任何效果的
    • r+  可以读,也可以写,文件打开的时候,指向文件开头,可以通过seek改变读写位置
    • w  这种方式打开的文件句柄,只能写,如果文件存在则将长度清零,否则新建文件,这种句柄通过seek之后,seek位置之前的文件数据全部变成0x00
    • w+ 同w选项,只不过多了一个可读功能
    • a  这种方式打开的文件,可以写,但是位置在文件末尾,即使往回seek也没有用,数据还是从文件末尾开始附加
    • a+ 同a选项,多了可读的功能

    另外还有2个选项,可以与上面的6个选项复合使用,一个是t表示以文本的方式打开文件(默认是t),一个是b表示以二进制的方式打开文件,t和b是互斥的不能同时使用。当与b组合时,有这么几种方式:wb、ab、rb、wb+、ab+、rb+,而a,w,r这几个选项是不能组合使用的,其中a,w都表示写文件,只不过一个在文件尾,一个在文件开始处,r表示读文件。我试过将a,w,r几个两两组合使用,发现下面的现象:

    • wr 与w效果一样
    • rw与r效果一样
    • aw与a效果一样
    • wa 与w效果一样
    • ar与a效果一样
    • ra与r效果一样

    可以看出来当a,w,r在一起组合使用的时候,其后面的选项实际上好像是被忽略了

    问题的解决方法:rb+打开文件

    所以解决文章开头提出来的问题,应该使用  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;
    }
    
  • 相关阅读:
    Windows远程桌面连接CentOS 7
    CentOS7 系统菜单中添加快捷方式
    CentOS 7 创建桌面快捷方式
    Centos7 可执行程序自定义为系统服务
    CentOS 7 中 Docker 的安装
    CentOS 7 需要安装的常用工具,及centos安装fcitx 搜狗输入法的坑旅
    C++中结构体与类的区别 2
    C++中结构体与类的区别 1
    C++ 11 自旋锁
    Imply.io单机安装
  • 原文地址:https://www.cnblogs.com/wangqiguo/p/4539268.html
Copyright © 2011-2022 走看看