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

原创
2016-06-23 14:27:43 1286浏览

我打开了一个文件(二进制)的,然后利用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这种软件都是内存映射的, 文件太大它也没办法, 只能告诉你失败.

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

声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn核实处理。