PHP打开一个二进制文件,修改了内容如何再保存回去呢?
下面是代码:
$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这种软件都是内存映射的, 文件太大它也没办法, 只能告诉你失败.
请问怎么可以调用保存窗口保存么