Android:解压缩连接的gzip压缩文件

时间:2015-02-25 09:01:50

标签: java android gzip

gzipdocumented以支持压缩文件的连接:

$ echo hello >hhh
$ echo world >www
$ cat hhh www
hello
world
$ echo hello | gzip >hhhh
$ echo world | gzip >wwww
$ cat hhhh wwww | gunzip
hello
world

我可以使用GZIPOutputStream创建连接文件,但不幸的是GZIPInputStream只读取数据的第一部分(从命令行运行的gunzip读取所有数据。)

我在Android 4.1.2和4.4.2上看到了这一点。

如何从Java读取整个文件?

更新:
演示错误(主机版本)的示例:

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.Arrays;
import java.util.zip.GZIPInputStream;
import java.util.zip.GZIPOutputStream;


class GZTest {
    static void append(File f, String s) {
        try {
            FileOutputStream fos = new FileOutputStream(f, true);
            //FileOutputStream gzos = fos;
            GZIPOutputStream gzos = new GZIPOutputStream(fos);
            gzos.write(s.getBytes("UTF-8"));
            gzos.close(); // TODO: do it finally{}
            fos.close(); // TODO: do it finally{}
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    static String readAll(File f) {
        try {
            FileInputStream fis = new FileInputStream(f);
            //FileInputStream gzis = fis;
            GZIPInputStream gzis = new GZIPInputStream(fis);
            byte[] buf = new byte[4096];
            int len = gzis.read(buf);
            gzis.close(); // TODO: do it finally{}
            fis.close(); // TODO: do it finally{}
            return new String(Arrays.copyOf(buf, len), "UTF-8");
        } catch (IOException e) {
            e.printStackTrace();
        }
        return null;
    }

}

public class A {
    public static void main(String[] args) {
        System.out.println("~~~");
        File f = new File("x.y");
        f.delete();
        GZTest.append(f, "Hello, ");
        GZTest.append(f, "world!\n");
        System.out.println(GZTest.readAll(f));
    }
}

运行它:

$ javac A.java
$ java A
~~~
Hello, 
$ gunzip <x.y
Hello, world!

UPDATE2
看起来这是JDK-2192186的错误,据报道将在2010-08-03修复。不过,这个bug现在就在这里。

2 个答案:

答案 0 :(得分:0)

对于主机java,正确的阅读方式是:

static String readAll(File f) {
    try {
        FileInputStream fis = new FileInputStream(f);
        //FileInputStream gzis = fis;
        GZIPInputStream gzis = new GZIPInputStream(fis);
        final int SIZE=4096;
        byte[] buf = new byte[SIZE];
        int len=0, read=0;
        do {
            read = gzis.read(buf, len, SIZE-len);
            if (read < 0) {
                break;
            }
            len += read;
        } while (len<SIZE);
        gzis.close(); // TODO: do it finally{}
        return new String(Arrays.copyOf(buf, len), "UTF-8");
    } catch (IOException e) {
        e.printStackTrace();
    }
    return null;
}

(此 readAll()将用于代替问题中的 readAll()

此代码不适用于Android !!!

在Android上,这仍然是

D/~~~     (30098): Hello, 

而命令行gunzip说

# cat /data/data/com.example.gzctest2/files/x.y | gunzip
Hello, world!

答案 1 :(得分:0)

刚刚尝试了truncated output from GZIPInputStream on Android的解决方案:它有效。 要使用修补程序https://github.com/ymnk/jzlib/tree/concatenated_gzip_streams克隆分支,请使用:

git clone https://github.com/ymnk/jzlib.git
git checkout concatenated_gzip_streams

然后将目录 src / main / java (您不能只复制一个文件)复制到项目中并替换导入:

//import java.util.zip.GZIPInputStream;
import com.jcraft.jzlib.GZIPInputStream;

适用于Android!

如果您希望删除不需要的文件需要: Adler32.java Deflate.java GZIPInputStream.java Inflate.java InfTree.java Tree.java Checksum.java GZIPException.java InfBlocks.java InflaterInputStream.java JZlib.java ZStream.java CRC32.java GZIPHeader.java InfCodes.java Inflater.java StaticTree.java
并且不会需要: Deflater.java DeflaterOutputStream.java GZIPOutputStream.java ZInputStream.java ZOutputStream.java ZStreamException.java

看起来删除6个未使用的文件(24K)而留下17个文件(198K)似乎不值得。