php は次のようにバイナリ ファイルを受け取ります:
header('Content-type: text/html; charset=utf-8');
$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;//ポストされたバイナリ生データを取得します
$file = fopen("./upload/".$filename,"w");// file 書き込みの準備
fwrite($file,$raw);//書き込み
fclose($file);//閉じる
?>
受信したバイナリ ファイル内の非標準文字 (>0x7F) のバイト数はそれを 3 バイトに置き換えます。たとえば、E2 を EF 9F A2 に置き換えます
次に、Php がファイルを受信した後にそれを復元し、EF 9F A2 を E2 に置き換えたいと思います。
ありがとうございました!
バイナリ ファイルではなくテキスト ファイルです
>0x7F はすべて utf-8 ワイド文字ではありませんか?
$raw の内容を投稿して見てみると良いでしょう
試してみてください
utf-8 の EF 9F A2 は必要ですE2 の代わりに Unicode F7E2
と E000-F8FF はユーザー定義文字に属します (つまり、ユーザーに任された領域以外に文字セットはありません)
たとえば、一部の通信インターフェースでは境界認識に外字を使用します
ここに注意してください
if (xmlhttp.overrideMimeType) {
xmlhttp.overrideMimeType('text/plain; charset=x-user-define') } else {
xmlhttp.setRequestHeader('Accept-Charset' , 'x-user-dependent');
}
xmlhttp.setRequestHeader("Content-Type","application/x-www-form-urlencoded;charset=utf-8") ;
ファイルの送信とファイルの受信:
送信しているものはバイナリ ストリームです
そうしないと、テキストとして送信するときに問題が発生します
送信時にバイナリデータとして宣言できるため、情報を確認する必要があります
オンラインでこのメソッドを見つけました:
function byteValue(x) {
return x.charCodeAt(0) & 0xff;
var ords = Array.prototype.map.call( datastr, byteValue);
var ui8a = new window.send(ui8a.buffer);
xmlhttp.sendAsBinary(コンテンツ);
PC で IE ブラウザを使用するだけです。
しかし、Android ブラウザで使用する必要があるため、この機能 Uint8Array をサポートしていないため、バイナリ ストリームを使用して送信する方法がわかりません。
送信されたファイルと受信されたファイルを比較すると、まだルールがあります:
C0 より大きいファイルのプレフィックスは EF 9F です (例: E2 は EF 9F A2 に置き換えられます)
C0 より小さいファイルのプレフィックスは EF 9E です (B2 など)は EF 9E B2 に置き換えられます
2進数を置き換える方法はありますか?
正規表現はサポートされていますか?
正規表現と同様に、str_replace と strtr の両方を使用できます
$s = str_replace("xEFx9ExB2", "xB2", $s); ただし、
が C0 より大きい場合、プレフィックスは次のようになります。 EF 9F、たとえば E2 は EF 9F に置き換えられます A2
彼のデータ変換ルールは次のとおりです:
ファイルヘッダーは変更されません。画像から 0050h までは同じです。
>=80 のバイトを変換し、上位ビットに F7 を追加して utf-8 に変換します。たとえば、E2 は F7E2 になり、utf-8 に変換されます。 EF 9F A2 になります
写真の少なくとも E2/FB/ 91/B2/81...すべてこのルールを満たします
彼が示しているのは sqlite データベース ファイルです。明らかに、バイト値 0x00 ~ 0xff を持つあらゆる種類のデータをデータベースに保存できます。
#10 が言ったように、それが正しいことが確認できたら、関数を作成します。そしてそれは完了します
アルゴリズムテスト
$s =<<< TXT00 2D EF 8F A2 1A 05 00 00 00 01 03 EF 9F BB 0000 00 00 0E 03 EF 9F BB 02 6B 01 EF 9E 91 02 3600 EF 9E B2 01 60 00 EF 9E 81 00 EF 9E 81 00 2FTXT;$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;}
str_replace、strtr
$s = str_replace("xEFx9ExB2", "xB2", $s); などの正規表現も使用できます
ただし、
が C0 より大きい場合は、次のようにプレフィックス EF 9F を追加します。 E2 を EF 9F A2 に置き換える
ルールは何ですか?
値が C0 より大きい場合、プレフィックスは EF 9F です。この正規表現をどのように記述するか?
データが変数 $s に格納されていると仮定すると、
$s = preg_replace_callback('/[\xef]../', 'foo', $s);function foo($r) { $c = (ord($r[0]{1}) & 0x03) << 6; $c += (ord($r[0]{2}) & 0x3f); return chr($c);}
00 2F
さらに、お聞きしたいのですが、この方法で変換された $c が文字形式で出力され、16 進数でファイルに書き戻された場合、どう対処すればよいでしょうか。
$raw[$j++]=$c;//raw を使用します
受け取った元のデータです
いいえ、もう一度変換する必要がありますか?
$raw を使用してファイルを直接書き込みたいです。
バイナリ データには、印刷できない文字が多く含まれており、直感的に確認できるように、通常は 1 バイトと 2 つの 16 進数の形式で印刷されます (マップ内のソフトウェアなど)
16 進数を変換するには、10 を変更します。文字列をバイナリデータに変換するには、通常、pack('H*', 'ffffff') を使用します
echo hex2bin("6578616d706c65206865782064617461"); // 16 進データの例
ありがとうございます、学びました