3

次のJavaコードを使用して、bytes[]をGZIPとの間で圧縮/解凍します。最初のテキストバイトからgzipバイト:

public static byte[] fromByteToGByte(byte[] bytes) {
    ByteArrayOutputStream baos = null;
    try {
        ByteArrayInputStream bais = new ByteArrayInputStream(bytes);
        baos = new ByteArrayOutputStream();
        GZIPOutputStream gzos = new GZIPOutputStream(baos);
        byte[] buffer = new byte[1024];
        int len;
        while((len = bais.read(buffer)) >= 0) {
            gzos.write(buffer, 0, len);
        }
        gzos.close();
        baos.close();
    } catch (IOException e) {
        e.printStackTrace();
    }
    return(baos.toByteArray());
}

次に、圧縮バイトから非圧縮バイトに逆方向に進むメソッド:

public static byte[] fromGByteToByte(byte[] gbytes) {
    ByteArrayOutputStream baos = null;
    ByteArrayInputStream bais = new ByteArrayInputStream(gbytes);
    try {
        baos = new ByteArrayOutputStream();
        GZIPInputStream gzis = new GZIPInputStream(bais);
        byte[] bytes = new byte[1024];
        int len;
        while((len = gzis.read(bytes)) > 0) {
            baos.write(bytes, 0, len);
        }
    } catch (IOException e) {
        e.printStackTrace();
    }
    return(baos.toByteArray());
}

gzipファイルに書き出していないので効果があると思いますか?
また、標準のC#関数では、BitConverterが最初の4バイトを読み取り、MemoryStream Write関数が開始点4、入力バッファー長-4で呼び出されることに気付きました。これはヘッダーの有効性に影響しますか?

ジム

4

2 に答える 2

8

試してみましたが、「無効なGZipヘッダー」の問題を再現できません。これが私がしたことです:

Java側

私はあなたのJava圧縮方法をこのJavaスニペットと一緒に取りました:

public static String ToHexString(byte[] bytes){
    StringBuilder hexString = new StringBuilder();
        for (int i = 0; i < bytes.length; i++)
            hexString.append((i == 0 ? "" : "-") + 
                Integer.toString((bytes[i] & 0xff) + 0x100, 16).substring(1));
    return hexString.toString();
}

そのため、この最小限のJavaアプリケーションは、テスト文字列のバイトを取得して圧縮し、圧縮データの16進文字列に変換します...:

public static void main(String[] args){
    System.out.println(ToHexString(fromByteToGByte("asdf".getBytes())));
}

...以下を出力します(注釈を追加しました)

1f-8b-08-00-00-00-00-00-00-00-4b-2c-4e-49-03-00-bd-f3-29-51-04-00-00-00
^------- GZip Header -------^ ^----------- Compressed data -----------^

C#側

バイト配列を別のバイト配列に圧縮および解凍するための2つのメソッドを作成しました(圧縮メソッドは完全を期すためのものであり、私のテストです)

public static byte[] Compress(byte[] uncompressed)
{
    using (MemoryStream ms = new MemoryStream())
    using (GZipStream gzs = new GZipStream(ms, CompressionMode.Compress))
    {
        gzs.Write(uncompressed, 0, uncompressed.Length);
        gzs.Close();
        return ms.ToArray();
    }
}

public static byte[] Decompress(byte[] compressed)
{
    byte[] buffer = new byte[4096];
    using (MemoryStream ms = new MemoryStream(compressed))
    using (GZipStream gzs = new GZipStream(ms, CompressionMode.Decompress))
    using (MemoryStream uncompressed = new MemoryStream())
    {
        for (int r = -1; r != 0; r = gzs.Read(buffer, 0, buffer.Length))
            if (r > 0) uncompressed.Write(buffer, 0, r);
        return uncompressed.ToArray();
    }
}

16進文字列を受け取り、それをバイト配列に戻す小さな関数と一緒に... (これもテスト目的で)

public static byte[] ToByteArray(string hexString)
{
    hexString = hexString.Replace("-", "");
    int NumberChars = hexString.Length;
    byte[] bytes = new byte[NumberChars / 2];
    for (int i = 0; i < NumberChars; i += 2)
        bytes[i / 2] = Convert.ToByte(hexString.Substring(i, 2), 16);
    return bytes;
}

...私は次のことをしました:

// Just hardcoded the output of the java program, convert it back to byte[]
byte[] fromjava = ToByteArray("1f-8b-08-00-00-00-00-00-00-00-" + 
                  "4b-2c-4e-49-03-00-bd-f3-29-51-04-00-00-00");

// Decompress it with my function above
byte[] uncompr = Decompress(fromjava);

// Get the string out of the byte[] and print it
Console.WriteLine(System.Text.ASCIIEncoding.ASCII
                    .GetString(uncompr, 0, uncompr.Length));

出来上がり、出力は次のとおりです。

asdf

私にぴったりです。たぶん、c#アプリケーションで解凍方法を確認する必要があります。

前の質問で、これらのバイト配列をデータベースに格納しているとおっしゃいましたね。たぶん、あなたはバイトがあなたがそれらを入れた方法でデータベースから戻ってくるかどうかをチェックしたいと思うでしょう。

于 2010-04-29T18:25:41.733 に答える
0

これを回答として投稿して、コードが適切に見えるようにします。いくつかの点に注意してください。
まず、データベースへのラウンドトリップは効果がないように見えました。両側のJavaは、私が入力したものを正確に生成しました。C#入力およびJava出力と同様に、C#出力のJavaはIonicAPIで正常に機能しました。それは私を2番目のポイントに導きます。第二に、私の元の解凍は次のオーダーでした:

public static string Decompress(byte[] gzBuffer)
{
    using (MemoryStream ms = new MemoryStream())
    {
        int msgLength = BitConverter.ToInt32(gzBuffer, 0);
        ms.Write(gzBuffer, 4, gzBuffer.Length – 4);
        byte[] buffer = new byte[msgLength];
        ms.Position = 0;
        using (GZipStream zip = new GZipStream(ms, CompressionMode.Decompress))
        {
            zip.Read(buffer, 0, buffer.Length);
        }
        return Encoding.UTF8.GetString(buffer);
    } 
}  

内部バイト数に依存しますが、内部値に関係なくファイル全体を読み取ります。Ionicアルゴリズムが何であるかわからない。あなたのメソッドは、私が使用したJavaメソッドと同じように機能します。それが私が見る唯一の違いです。そのすべての仕事をしてくれてありがとう。そのやり方を覚えています。ありがとう、ジム

于 2010-04-29T19:36:27.623 に答える