compressedoutputstream.java
来自「kaffe Java 解释器语言,源码,Java的子集系统,开放源代码」· Java 代码 · 共 197 行
JAVA
197 行
/* * $Id: CompressedOutputStream.java,v 1.1 2004/01/10 23:34:31 dalibor Exp $ * Copyright (C) 2003 The Free Software Foundation * * This file is part of GNU inetlib, a library. * * GNU inetlib is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * GNU inetlib is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * As a special exception, if you link this library with other files to * produce an executable, this library does not by itself cause the * resulting executable to be covered by the GNU General Public License. * This exception does not however invalidate any other reasons why the * executable file might be covered by the GNU General Public License. */package gnu.inet.ftp;import java.io.FilterOutputStream;import java.io.IOException;import java.io.OutputStream;/** * A DTP output stream that implements the FTP compressed transfer mode. * * @author <a href='mailto:dog@gnu.org'>Chris Burdess</a> * @version $Revision: 1.1 $ $Date: 2004/01/10 23:34:31 $ */class CompressedOutputStream extends DTPOutputStream{ static final byte RECORD = -128; // 0x80 static final byte EOF = 64; // 0x40 CompressedOutputStream(DTP dtp, OutputStream out) { super(dtp, out); } /** * Just one byte cannot be compressed. * It takes 5 bytes to transmit - hardly very compressed! */ public void write(int c) throws IOException { if (transferComplete) return; byte[] buf = new byte[] { RECORD, /* record descriptor */ 0x00, 0x01, /* one byte */ 0x01, /* one uncompressed byte */ (byte) c /* the byte */ }; out.write(buf, 0, 5); } public void write(byte[] b) throws IOException { write(b, 0, b.length); } /** * The larger len is, the better. */ public void write(byte[] b, int off, int len) throws IOException { if (transferComplete) return; byte[] buf = compress(b, off, len); len = buf.length; buf[0] = RECORD; /* record descriptor */ buf[1] = (byte) ((len & 0x00ff) >> 8); /* high byte of bytecount */ buf[2] = (byte) (len & 0xff00); /* low byte of bytecount */ out.write(buf, 0, len); } /** * Returns the compressed form of the given byte array. * The first 3 bytes are left free for header information. */ byte[] compress(byte[] b, int off, int len) { byte[] buf = new byte[len]; byte last = 0; int pos = 0, raw_count = 0, rep_count = 1; for (int i = off; i < len; i++) { byte c = b[i]; if (i > off && c == last) // compress { if (raw_count > 0) // flush raw bytes to buf { // need to add raw_count+1 bytes if (pos + (raw_count + 1) > buf.length) buf = realloc(buf, len); pos = flush_raw(buf, pos, b, (i - raw_count) - 1, raw_count); raw_count = 0; } rep_count++; // keep looking for same byte } else { if (rep_count > 1) // flush compressed bytes to buf { // need to add 2 bytes if (pos + 2 > buf.length) buf = realloc(buf, len); pos = flush_compressed(buf, pos, rep_count, last); rep_count = 1; } raw_count++; // keep looking for raw bytes } if (rep_count == 127) // flush compressed bytes { // need to add 2 bytes if (pos + 2 > buf.length) buf = realloc(buf, len); pos = flush_compressed(buf, pos, rep_count, last); rep_count = 1; } if (raw_count == 127) // flush raw bytes { // need to add raw_count+1 bytes if (pos + (raw_count + 1) > buf.length) buf = realloc(buf, len); pos = flush_raw(buf, pos, b, (i - raw_count), raw_count); raw_count = 0; } last = c; } if (rep_count > 1) // flush compressed bytes { // need to add 2 bytes if (pos + 2 > buf.length) buf = realloc(buf, len); pos = flush_compressed(buf, pos, rep_count, last); rep_count = 1; } if (raw_count > 0) // flush raw bytes { // need to add raw_count+1 bytes if (pos + (raw_count + 1) > buf.length) buf = realloc(buf, len); pos = flush_raw(buf, pos, b, (len - raw_count), raw_count); raw_count = 0; } byte[] ret = new byte[pos + 3]; System.arraycopy(buf, 0, ret, 3, pos); return ret; } int flush_compressed(byte[] buf, int pos, int count, byte c) { buf[pos++] = (byte) (0x80 | count); buf[pos++] = c; return pos; } int flush_raw(byte[] buf, int pos, byte[] src, int off, int len) { buf[pos++] = (byte) len; System.arraycopy(src, off, buf, pos, len); return pos + len; } byte[] realloc(byte[] buf, int len) { byte[] ret = new byte[buf.length + len]; System.arraycopy(buf, 0, ret, 0, buf.length); return ret; } public void close() throws IOException { byte[] buf = new byte[] { EOF, /* eof descriptor */ 0x00, 0x00 /* no bytes */ }; out.write(buf, 0, 3); out.close(); }}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?