php接收二进制文件如下:
<?php
header('Content-type: text/html; charset=utf-8'); 
error_reporting(0); 
$filename = $_GET["filename"];
$filesize = $_GET["filesize"];$xmlstr =  $GLOBALS[HTTP_RAW_POST_DATA];//$_POST["data"];//
if(empty($xmlstr)) $xmlstr = file_get_contents('php://input');$raw = $xmlstr;//得到post过来的二进制原始数据
$file = fopen("./upload/".$filename,"w");//打开文件准备写入
fwrite($file,$raw);//写入
fclose($file);//关闭
?>接收的二进制文件中非标准字符(>0x7F)的字节被替换为三个字节,比如E2替换为EF 9F A2
现在我想在Php接收到文件之后还原回去,把EF 9F A2替换为E2,请问这个怎么实现?非常感谢!
PHP二进制

解决方案 »

  1.   

    你收到的不是一个 XML 流吗?
    是文本文件而非二进制文件呀
    >0x7F 的不都是 utf-8 宽字符吗?你最好贴出 $raw 的内容看看
      

  2.   

    fopen("./upload/".$filename,"wb");
    试试
      

  3.   

    补充
    utf-8的EF 9F A2 应该是unicode F7E2 而不是 E2
    且E000-F8FF属于用户自定义字符(就是没有字符集以外,交给用户使用的区域)
    举例就是某些通信接口用自定义字符来做边界识别要留意这个
      

  4.   

    发送端,格式是这样设置的:
    if (xmlhttp.overrideMimeType) {
    xmlhttp.overrideMimeType('text/plain; charset=x-user-defined');
    } else {
    xmlhttp.setRequestHeader('Accept-Charset', 'x-user-defined');
    }
    xmlhttp.setRequestHeader("Content-Type","application/x-www-form-urlencoded;charset=utf-8");发送文件和接收文件:
      

  5.   

    你传送的是二进制流
    按 XML 的约定,应以 base64 编码进行传送的
    否则当做文本传送就会有问题的传送时是否可以声明为二进制数据,这个需要查资料
      

  6.   

    在网上找到这种方法:
    XMLHttpRequest.prototype.sendAsBinary = function(datastr) {
    function byteValue(x) {
    return x.charCodeAt(0) & 0xff;
    }
    var ords = Array.prototype.map.call(datastr, byteValue);
    var ui8a = new window.Uint8Array(ords);
    this.send(ui8a.buffer);
     }
    xmlhttp.sendAsBinary(content);在PC上用IE浏览器就可以。
    但是我需要用在android 浏览器上,不支持这个函数Uint8Array,所以不知道怎么用二进制流发送了。
      

  7.   

    比较发送文件和接收文件,还是有规律的:
    大于C0的加前缀为EF 9F,如E2替换为EF 9F A2
    小于C0的加前缀为EF 9E,如B2替换为EF 9E B2这样的编码不知道是怎么出来的?所以想说对接收文件的十六进制做替换,只要结果正确就行了。
      

  8.   

    str_replace、strtr 都可以,正则也可以

    $s = str_replace("\xEF\x9E\xB2", "\xB2", $s);不过
    大于C0的加前缀为EF 9F,如E2替换为EF 9F A2
    的规则是什么?
      

  9.   

    他的数据转换规则是这样的:
    文件头不变,文件头多长不清楚,图上到0050h都一样的
    <=7F的字节,因为utf-8也一样,可以认为也转换了utf-8
    >=80的字节,高位加上F7,再转utf-8,例如E2就变成F7E2然后转utf-8成为EF 9F A2至少他图中E2/FB/91/B2/81...都满足这个规则
      

  10.   

    他图示的是 sqlite 数据库文件,显然数据库中是什么样数据都可以存放的,字节值 0x00 - 0xff
    若能证实确如 #10 所说,那么写个函数就搞定了
    算法测试$s =<<< TXT
    00 2D EF 8F A2 1A 05 00 00 00 01 03 EF 9F BB 00
    00 00 00 0E 03 EF 9F BB 02 6B 01 EF 9E 91 02 36
    00 EF 9E B2 01 60 00 EF 9E 81 00 EF 9E 81 00 2F
    TXT;
    $s = preg_replace("/[ \r\n]/", '', $s);
    $s = pack('H*', $s); //构造出二进制数据$m = 0;
    for($i=0; $i<strlen($s); $i++) {
      $c = ord($s{$i});
      if($c == 0xEF) {
        $c = $c & 0x0f;
        $c = ($c << 6) + (ord($s{++$i}) & 0x3f);
        $c = ($c << 6) + (ord($s{++$i}) & 0x3f);
        $c &= 0xff;
      }
      printf('%02X ', $c);
      if(($m = ($m+1) % 16) == 0) echo PHP_EOL;
    }
    00 2D E2 1A 05 00 00 00 01 03 FB 00 00 00 00 0E 
    03 FB 02 6B 01 91 02 36 00 B2 01 60 00 81 00 81 
    00 2F 
      

  11.   


    大于C0的加前缀为EF 9F,将EF 9F A2替换为A2+40,这个正则表达式怎么写呢?
      

  12.   

    本帖最后由 xuzuning 于 2013-05-02 15:35:02 编辑
      

  13.   


    另外请教下,这样转换的$c是字符形式打印出来,如果以十六进制写回文件,要怎么处理?
    我用
    $raw[$j++]=$c;//raw是接收到的原始数据
    不行,应该还要再转换一下吗?
    我是想用$raw直接写文件的。
      

  14.   

    二进制数据中包含了很多不可打印的字符,为了直观的看到他们,一般将其以一个字节两个十六进制数的格式打印出来(比如你贴图中的软件)将十六进制串转换成二进制数据,一般可用 pack('H*', 'ffffff')
    php5.4 还提供了一个 hex2bin
    echo hex2bin("6578616d706c65206865782064617461"); //example hex data