Android 压缩字符串

Android 端可以对字符串进行压缩。 在进行大量简单文本传输时,可以先压缩字符串再发送。接收端接收后再解压。也可以将字符串压缩后存入数据库中。

使用到的类库

  • GZIPInputStream
  • GZIPOutputStream

代码示例

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.util.zip.GZIPInputStream;
import java.util.zip.GZIPOutputStream;
​
public class StrZipUtil {
​
 /**
 * @param input 需要压缩的字符串
 * @return 压缩后的字符串
 * @throws IOException IO
 */
 public static String compress(String input) throws IOException {
 if (input == null || input.length() == 0) {
 return input;
 }
 ByteArrayOutputStream out = new ByteArrayOutputStream();
 GZIPOutputStream gzipOs = new GZIPOutputStream(out);
 gzipOs.write(input.getBytes());
 gzipOs.close();
 return out.toString("ISO-8859-1");
 }
​
 /**
 * @param zippedStr 压缩后的字符串
 * @return 解压缩后的
 * @throws IOException IO
 */
 public static String uncompress(String zippedStr) throws IOException {
 if (zippedStr == null || zippedStr.length() == 0) {
 return zippedStr;
 }
 ByteArrayOutputStream out = new ByteArrayOutputStream();
 ByteArrayInputStream in = new ByteArrayInputStream(zippedStr
 .getBytes("ISO-8859-1"));
 GZIPInputStream gzipIs = new GZIPInputStream(in);
 byte[] buffer = new byte[256];
 int n;
 while ((n = gzipIs.read(buffer)) >= 0) {
 out.write(buffer, 0, n);
 }
 // toString()使用平台默认编码,也可以显式的指定如toString("GBK")
 return out.toString();
 }
}

红米手机测试输出

08-09 13:16:53.388 32248-32267/com.rustfisher.ndkproj D/rustApp: 开始存入数据库 ori1 len=304304
08-09 13:16:53.418 32248-32267/com.rustfisher.ndkproj D/rustApp: 已存入数据库  ori1 len=304304 , 耗时约37 ms
08-09 13:16:53.418 32248-32267/com.rustfisher.ndkproj D/rustApp: 开始压缩  ori1 len=304304
08-09 13:16:53.438 32248-32267/com.rustfisher.ndkproj D/rustApp: 压缩完毕  zip1 len=1112 , 耗时约19 ms
08-09 13:16:53.438 32248-32267/com.rustfisher.ndkproj D/rustApp: 存压缩后的数据进数据库 zip1.length=1112
08-09 13:16:53.448 32248-32267/com.rustfisher.ndkproj D/rustApp: 压缩后的数据已进数据库 zip1.length=1112 , 耗时约8 ms
08-09 13:16:53.448 32248-32267/com.rustfisher.ndkproj D/rustApp: 解压开始
08-09 13:16:53.488 32248-32267/com.rustfisher.ndkproj D/rustApp: 解压完毕 耗时约36 ms

存储时间受存储字符串的长度影响。字符串长度与存储耗时正相关。

荣耀手机测试

08-09 10:38:42.759 23075-23109/com.rustfisher D/rustApp: 开始压缩  ori1 len=304304
08-09 10:38:42.764 23075-23109/com.rustfisher D/rustApp: 压缩完毕  zip1 len=1112
08-09 10:38:42.764 23075-23109/com.rustfisher D/rustApp: 解压开始
08-09 10:38:42.789 23075-23109/com.rustfisher D/rustApp: 解压完毕

此例中,荣耀压缩耗时约5ms,解压耗时约25ms。

可以看出,压缩后与原长度之比 1112/304304, 约0.365% 压缩和解压缩耗时视手机情况而定。

使用 ZipFile进行解压

使用Kotlin实现

Android中使用 ZipFile 与相关类来进行文件解压操作。

  • java.util.zip.ZipEntry 描述zip中的文件
  • java.util.zip.ZipFile 描述zip文件
  • java.util.zip.ZipInputStream 包含ZipEntry信息

1,解压assets里的zip

本例处理 assets 中的zip文件。例子中解压得到一份网页文件。

ZipFile 需要一个 File 对象。而assets中的文件不能直接当成 File 来用。 第一步我们先把目标 zip 复制一份出来。以下是执行复制操作的代码。

val tempFile = File(targetLocation, "tmp-$assetsZipName.zip")
try {
 val inputStream = assets.open(assetsZipName)
 if (tempFile.exists()) {
 tempFile.delete()
 }
 tempFile.createNewFile()
 val copyOs: OutputStream = FileOutputStream(tempFile)
 val tmp = ByteArray(1024)
 var len: Int
 while (((inputStream.read(tmp)).also { len = it }) != -1) {
 copyOs.write(tmp, 0, len)
 }
 copyOs.flush()
 copyOs.close()
 inputStream.close()
 Log.d(TAG, "临时文件复制完毕")
} catch (e: Exception) {
 Log.e(TAG, "unzipAssetsFile: ", e)
 return
}

如果文件存放在 app 内部存储或者 SD 卡,就不用这么麻烦。

得到临时 zip 后,使用临时 zip 来进行解压操作。

通过文件得到 ZipInputStream ,它带有压缩文件内部的各个文件的信息。用 ZipEntry 来描述。

拿到 ZipFile 对象,用 getInputStream(entry) 方法,得到每一个文件(目录)的输入流。然后用流来复制各个压缩的文件。

val zipInputStream = ZipInputStream(FileInputStream(tempFile))
val zipFile = ZipFile(tempFile)
var entry: ZipEntry?
​
while (zipInputStream.nextEntry.also { entry = it } != null) {
 val outFile = File(targetLocation, entry!!.name)
 Log.d(TAG, "当前文件: $entry -> $outFile")
 if (outFile.parentFile != null && !outFile.parentFile!!.exists()) {
 outFile.parentFile!!.mkdir()
 }
​
 if (!outFile.exists()) {
 if (entry!!.isDirectory) {
 outFile.mkdirs()
 continue
 } else {
 outFile.createNewFile()
 }
 }
​
 val bis = BufferedInputStream(zipFile.getInputStream(entry))
 val bos = BufferedOutputStream(FileOutputStream(outFile))
 val entryTmpArr = ByteArray(1024)
 while (true) {
 val readLen = bis.read(entryTmpArr)
 if (readLen == -1) {
 break
 }
 bos.write(entryTmpArr, 0, readLen)
 }
 bos.close()
 bis.close()
 Log.d(TAG, "解压得到文件 $outFile")
}

整个方法代码如下

/**
 * 解压assets里指定的某个zip
 */
private fun unzipAssetsFile(assetsZipName: String, targetLocation: String) {
 Log.d(TAG, "[unzipAssetsFile] targetLocation: $targetLocation")
 val targetDir = File(targetLocation)
 if (!targetDir.exists()) {
 targetDir.mkdirs()
 }
 val tempFile = File(targetLocation, "tmp-$assetsZipName.zip")
 try {
 val inputStream = assets.open(assetsZipName)
 if (tempFile.exists()) {
 tempFile.delete()
 }
 tempFile.createNewFile()
 val copyOs: OutputStream = FileOutputStream(tempFile)
 val tmp = ByteArray(1024)
 var len: Int
 while (((inputStream.read(tmp)).also { len = it }) != -1) {
 copyOs.write(tmp, 0, len)
 }
 copyOs.flush()
 copyOs.close()
 inputStream.close()
 Log.d(TAG, "临时文件复制完毕")
 } catch (e: Exception) {
 Log.e(TAG, "unzipAssetsFile: ", e)
 return
 }
​
 val zipInputStream = ZipInputStream(FileInputStream(tempFile))
 val zipFile = ZipFile(tempFile)
 var entry: ZipEntry?
​
 while (zipInputStream.nextEntry.also { entry = it } != null) {
 val outFile = File(targetLocation, entry!!.name)
 Log.d(TAG, "当前文件: $entry -> $outFile")
 if (outFile.parentFile != null && !outFile.parentFile!!.exists()) {
 outFile.parentFile!!.mkdir()
 }
​
 if (!outFile.exists()) {
 if (entry!!.isDirectory) {
 outFile.mkdirs()
 continue
 } else {
 outFile.createNewFile()
 }
 }
​
 val bis = BufferedInputStream(zipFile.getInputStream(entry))
 val bos = BufferedOutputStream(FileOutputStream(outFile))
 val entryTmpArr = ByteArray(1024)
 while (true) {
 val readLen = bis.read(entryTmpArr)
 if (readLen == -1) {
 break
 }
 bos.write(entryTmpArr, 0, readLen)
 }
 bos.close()
 bis.close()
 Log.d(TAG, "解压得到文件 $outFile")
 }
 val delTmp = tempFile.delete()
 Log.d(TAG, "解压完毕 删除临时文件$delTmp")
}

2,电脑压缩 zip

mac上如果在 Finder 里进行压缩,可能在操作的时候,系统创建了一个 __MACOS的目录。 为了避免这个目录出现。我们可以用 zip 命令来压缩。

zip -r target.zip sourceDir

Android 零基础入门教程视频参考