欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  后端开发

PHP打开一个二进制文件,修改了内容如何再保存回去呢?

程序员文章站 2022-04-15 10:02:42
...
我打开了一个文件(二进制)的,然后利用bin2hex函数,将其变成了十六进制的字符串,并且修改了其中一个特定位置的数值,现在问题是,我想保存回去,发现变成了ASCII文件,请问各位,如何应该保存成和源文件格式一样的代码呢?

下面是代码:


$fp=fopen("sample.bin",'rb');
$word='';
while(!feof($fp)){
$buf = bin2hex(fread($fp,48));
$pos = strpos($buf, '02000100'); // 查找土地

//if $pos == 0 {
// echo "这是土地";
//}
echo $buf;
echo "\n-------------\n";
echo ($pos/2);

echo $buf2=str_replace("02000100","BE010000",$buf);
$word=$word.$buf2;

echo "\n";


}
fclose($fp);

echo $word;

$new= fopen('new.bin','w');
fwrite($new,$word);
fclose($new);
?>

回复讨论(解决方案)

一个文件一种格式,存两种格式也没有意义,可以在需要时候再取出来转换,不用转为十六进制

1、既然你用 $fp=fopen("sample.bin",' rb'); 打开文件,表示你的程序可在 window 系统下工作,那么写入时也应 $new= fopen('new.bin','w b');
b 表示二进制方式,window 下必须严格区别,linux 下无所谓

2、处理后的 $word 的内容是十六进制表示,你在写入文件时并没有再转回二进制
可用 $word = pack('H*', $word); 转换

$word = '';
$from = pack('H*', '02000100');
$target = pack('H*', 'BE010000');
while(!feof($fp)){
$buf = bin2hex(fread($fp,48));
$buf2 = str_replace($from, $target, $buf);
$word .= $buf2;
}

这样 $word 中依旧是二进制数据

$word = '';
$from = pack('H*', '02000100');
$target = pack('H*', 'BE010000');
while(!feof($fp)){
$buf = bin2hex(fread($fp,48));
$buf2 = str_replace($from, $target, $buf);
$word .= $buf2;
……

谢谢xuzuning的意见,我按照你的修改了代码,如下:


$fp=fopen("old.bin",'rb');
$word='';
$from = pack('H*', '02000100');
$target = pack('H*', 'BE010000');
while(!feof($fp)){
$buf = bin2hex(fread($fp,48));
$buf2 = str_replace($from, $target, $buf);
$pos = strpos($buf, '02000100'); // 查找土地

//if $pos == 0 {
// echo "这是土地";
//}
echo $buf;
echo "\n-------------\n";
echo ($pos/2);
echo $buf2;
//echo $buf2=str_replace("02000100","BE010000",$buf);
$word=$word.$buf2;

echo "\n";


}
fclose($fp);

//echo $word;

$new= fopen('new.bin','wb');
fwrite($new,$word);
fclose($new);
?>

但是生成的new.bin的大小依然是old.bin的两倍,和之前没有什么区别,我的测试环境是在我的WIN2008的服务器上,IIS7+PHP,您还有什么建议么?

对不起,我 #3 的代码错了,应该是这样
$word = '';
$from = pack('H*', '02000100');
$target = pack('H*', 'BE010000');
while(!feof($fp)){
$buf = fread($fp,48);
$buf2 = str_replace($from, $target, $buf);
$word .= $buf2;
}

完全按二进制方式操作

对不起,我 #3 的代码错了,应该是这样
$word = '';
$from = pack('H*', '02000100');
$target = pack('H*', 'BE010000');
while(!feof($fp)){
$buf = fread($fp,48);
$buf2 = str_replace($from, $target, $buf);
$wo……

感谢,问题解决!

唠叨说了关键了, windows的确区分文本和二进制文件, 因为文本的末尾是以某个特殊字节标示的.

WINDOWS下,不太了解的同学经常用fgetc函数以文本打开二进制文件, 试图一个字节一个字节的读取, 结果发现只能读了一半就开始返回EOF了, 其实是因为恰好某个字节是文本里的作为结束符的特殊字节。

但Linux一律以系统API : read返回0作为文件末,所以b参数是被忽略的。

楼主的作法多余了, 不就是想找到 土地 两个字, 然后替换一下吗 ?
不知道楼主所说的二进制文件是怎么生成的, 其实一切都是二进制, 只是在windows下b打开和非b打开得到的结果会有区别, 非b打开和用文本编辑器编辑结果是一致的.

假设你原本的文件就是一个utf8文本文件,里面有“土地”两个字,作法很简单,UTF8多字节编码的,循环fread并不一定正好能将完整的“土地”两个字的字节读到一个buffer里,所以比较好的办法是file_get_contents,之后直接str_replace("土地", “兄弟”)即, 在C语言里其实就是内存映射后替换字符串.

转什么十六进制, 纯属多此一举, 而且循环读是一定有bug的, 可能"土地"两个字是被拆开读进来的.

就连vim这种软件都是内存映射的, 文件太大它也没办法, 只能告诉你失败.

请问怎么可以调用保存窗口保存么