Сжатие Zlib с использованием классов Deflate и Inflate в Java
Я хочу попытаться использовать классы Deflate и Inflate в java.утиль.zip для сжатия zlib.
Я могу сжать код, используя Deflate, но при распаковке у меня возникает эта ошибка -
Exception in thread "main" java.util.zip.DataFormatException: unknown compression method
at java.util.zip.Inflater.inflateBytes(Native Method)
at java.util.zip.Inflater.inflate(Inflater.java:238)
at java.util.zip.Inflater.inflate(Inflater.java:256)
at zlibCompression.main(zlibCompression.java:53)
вот мой код до сих пор -
import java.util.zip.*;
import java.io.*;
public class zlibCompression {
/**
* @param args
*/
public static void main(String[] args) throws IOException, DataFormatException {
// TODO Auto-generated method stub
String fname = "book1";
FileReader infile = new FileReader(fname);
BufferedReader in = new BufferedReader(infile);
FileOutputStream out = new FileOutputStream("book1out.dfl");
//BufferedInputStream in = new BufferedInputStream(new FileInputStream(filename));
Deflater compress = new Deflater();
Inflater decompress = new Inflater();
String readFile = in.readLine();
byte[] bx = readFile.getBytes();
while(readFile!=null){
byte[] input = readFile.getBytes();
byte[] compressedData = new byte[1024];
compress.setInput(input);
compress.finish();
int compressLength = compress.deflate(compressedData, 0, compressedData.length);
//System.out.println(compressedData);
out.write(compressedData, 0, compressLength);
readFile = in.readLine();
}
File abc = new File("book1out.dfl");
InputStream is = new FileInputStream("book1out.dfl");
InflaterInputStream infl = new InflaterInputStream(new FileInputStream("book1out.dfl"), new Inflater());
FileOutputStream outFile = new FileOutputStream("decompressed.txt");
byte[] b = new byte[1024];
while(true){
int a = infl.read(b,0,1024);
if(a==0)
break;
decompress.setInput(b);
byte[] fresult = new byte[1024];
//decompress.in
int resLength = decompress.inflate(fresult);
//outFile.write(b,0,1);
//String outt = new String(fresult, 0, resLength);
//System.out.println(outt);
}
System.out.println("complete");
}
}
2 ответов
что вы пытаетесь здесь сделать? Вы используете InflaterInputStream, который распаковывает ваши данные, а затем вы пытаетесь передать эти распакованные данные снова в Inflater? Используйте любой из них, но не оба.
Это то, что вызывает исключение.
В дополнение к этому, есть довольно некоторые незначительные ошибки, такие как упомянутые bestsss:
- вы заканчиваете обжатие в петле-после заканчивать, никакие больше данных не могут быть добавлен.
- вы не проверяете, сколько выхода производит процесс дефляции. Если у вас длинные строки, это может быть более 1024 байт.
- вы устанавливаете входной сигнал к Inflater без устанавливать длину
a
тоже.
еще несколько, которые я нашел:
- вы не закрываете FileOutputStream после записи (и перед чтением из того же файла).
- вы используете
readLine()
для чтения строки текста, но затем вы не добавляете строку перерыв снова, что означает, что в вашем распакованном файле не будет никаких разрывов строк. - преобразование из байтов в строку и Байт, снова без необходимости.
- вы создаете переменные, которые вы не используете позже.
Я не буду пытаться исправить вашу программу. Вот простой, который делает то, что я думаю, вы хотите, используя DeflaterOutputStream и InflaterInputStream. (Вы также можете использовать ZInputStream и ZOutputStream JZlib вместо.)
import java.util.zip.*;
import java.io.*;
/**
* Example program to demonstrate how to use zlib compression with
* Java.
* Inspired by http://stackoverflow.com/q/6173920/600500.
*/
public class ZlibCompression {
/**
* Compresses a file with zlib compression.
*/
public static void compressFile(File raw, File compressed)
throws IOException
{
InputStream in = new FileInputStream(raw);
OutputStream out =
new DeflaterOutputStream(new FileOutputStream(compressed));
shovelInToOut(in, out);
in.close();
out.close();
}
/**
* Decompresses a zlib compressed file.
*/
public static void decompressFile(File compressed, File raw)
throws IOException
{
InputStream in =
new InflaterInputStream(new FileInputStream(compressed));
OutputStream out = new FileOutputStream(raw);
shovelInToOut(in, out);
in.close();
out.close();
}
/**
* Shovels all data from an input stream to an output stream.
*/
private static void shovelInToOut(InputStream in, OutputStream out)
throws IOException
{
byte[] buffer = new byte[1000];
int len;
while((len = in.read(buffer)) > 0) {
out.write(buffer, 0, len);
}
}
/**
* Main method to test it all.
*/
public static void main(String[] args) throws IOException, DataFormatException {
File compressed = new File("book1out.dfl");
compressFile(new File("book1"), compressed);
decompressFile(compressed, new File("decompressed.txt"));
}
}
для большей эффективности может быть полезно обернуть потоки файлов буферизованными потоками. Если это критическая производительность, измерьте ее.
Paŭlo Ebermannкод может быть дополнительно улучшен с помощью try-with-resources:
import java.util.Scanner;
import java.util.zip.*;
import java.io.*;
public class ZLibCompression
{
public static void compress(File raw, File compressed) throws IOException
{
try (InputStream inputStream = new FileInputStream(raw);
OutputStream outputStream = new DeflaterOutputStream(new FileOutputStream(compressed)))
{
copy(inputStream, outputStream);
}
}
public static void decompress(File compressed, File raw)
throws IOException
{
try (InputStream inputStream = new InflaterInputStream(new FileInputStream(compressed));
OutputStream outputStream = new FileOutputStream(raw))
{
copy(inputStream, outputStream);
}
}
public static String decompress(File compressed) throws IOException
{
try (InputStream inputStream = new InflaterInputStream(new FileInputStream(compressed)))
{
return toString(inputStream);
}
}
private static String toString(InputStream inputStream)
{
try (Scanner scanner = new Scanner(inputStream).useDelimiter("\A"))
{
return scanner.hasNext() ? scanner.next() : "";
}
}
private static void copy(InputStream inputStream, OutputStream outputStream)
throws IOException
{
byte[] buffer = new byte[1000];
int length;
while ((length = inputStream.read(buffer)) > 0)
{
outputStream.write(buffer, 0, length);
}
}
}