androidAndroid技术知识Android知识

压缩 json 的一些尝试

2017-06-08  本文已影响262人  sunrain_

前言

因为自己维护着一款课程表软件,软件里有通过二维码分享课程的功能,大致实现就是将课程信息转换为 json 放入二维码中,扫描后解析 json 还原课程信息。

但有时课程表内有太多课程,导致 json 过长,生成的二维码过于密集,不好扫描,于是便有了这次压缩 json 的尝试。

缩短 Key

json 是 key-value 结构,如果定义好规范,则可以将 key 尽量缩短,甚至是无意义的字母,但前提是文档一定要写清楚,避免不必要的麻烦。

我在项目中使用了 abcd... 来代替 key,缩短了一半的长度:

[
 {"h":"华罗庚楼","g":"数学","f":0,"e":2,"b":0,"d":0,"c":1,"a":1},
 {"h":"华盛顿楼","g":"英语","f":0,"e":15,"b":1,"d":0,"c":0,"a":3},
 {"h":"闻一多楼","g":"语文","f":0,"e":13,"b":1,"d":0,"c":0,"a":1},
 {"h":"居里夫人楼","g":"化学","f":0,"e":6,"b":0,"d":0,"c":0,"a":3},
 {"h":"华罗庚楼","g":"数学","f":0,"e":2,"b":0,"d":0,"c":1,"a":4},
 {"h":"闻一多楼","g":"语文","f":0,"e":13,"b":0,"d":0,"c":0,"a":0},
 {"h":"伽利略楼","g":"物理","f":0,"e":19,"b":0,"d":0,"c":1,"a":2},
 {"h":"伽利略楼","g":"物理","f":0,"e":19,"b":1,"d":0,"c":0,"a":0}
]

如果使用的是 org.json.JSONObject,直接放入键值对就行,比较方便。
而我的项目中使用的是 Gson 2.8.0 版本,想要实现需要额外的代码。

首先在属性上加上注解,举个例子:

@Expose //没有此注解的属性不会被序列化
@SerializedName("a") //序列化后的名称
public int week;

然后创建 Gson 对象并序列化:

Gson gson = new GsonBuilder().excludeFieldsWithoutExposeAnnotation().create();
Type type = new TypeToken<List<ClassBean>>() {}.getType();
return gson.fromJson(json, type);

这里举例的 type 为 List<ClassBean>,请根据实际情况替换。
这样就能得到和上边一样的 json 了。

Deflater & Inflater

Deflater 是同时使用了LZ77算法与哈夫曼编码的一个无损数据压缩算法。

LZ77算法图解(转载自Bugly)

我们可以使用 java 提供的 Deflater 和 Inflater 类对 json 进行压缩和解压缩:

// 压缩
public static String zipString(String unzip) {
    Deflater deflater = new Deflater(9); // 0 ~ 9 压缩等级 低到高
    deflater.setInput(unzip.getBytes());
    deflater.finish();

    final byte[] bytes = new byte[256];
    ByteArrayOutputStream outputStream = new ByteArrayOutputStream(256);

    while (!deflater.finished()) {
        int length = deflater.deflate(bytes);
        outputStream.write(bytes, 0, length);
    }

    deflater.end();
    return Base64.encodeToString(outputStream.toByteArray(), Base64.NO_PADDING);
}
// 解压缩
@Nullable
public static String unzipString(String zip) {
    byte[] decode = Base64.decode(zip, Base64.NO_PADDING);

    Inflater inflater = new Inflater();
    inflater.setInput(decode);

    final byte[] bytes = new byte[256];
    ByteArrayOutputStream outputStream = new ByteArrayOutputStream(256);

    try {
        while (!inflater.finished()) {
            int length = inflater.inflate(bytes);
            outputStream.write(bytes, 0, length);
        }
    } catch (DataFormatException e) {
        e.printStackTrace();
        return null;
    } finally {
        inflater.end();
    }

    return outputStream.toString();
}

压缩过后的 json 又缩小了 44%:

eNqLrlbKULJSetrb93zv9Ke7Zj1bukdJRykdKPRs6oana5cBOWlKVgY6SqlKVkY6SklgdgqYTFayMtRRSgSSt
ToIU2bPfrlwP8KUF90bX6xfi2yKoSnYGEMkYwzAxhjDjHk5ffeTHQ1PlyA5BmjGs2ntKMYY4zAG4ZqNrS/be5
4uWf1k1y6ESU97pqF5ywzDW2juoShwTMjxFaZjDGDGPNmz92nHyudTlyKMed658vmENhRjLHG4xogcYwyxuCY
WANlUxMk

二维码对比

压缩前 压缩后

可以看到二维码的内容已经少了很多,这次的优化就到此为止了。
欢迎留言探讨,谢谢大家。

上一篇下一篇

猜你喜欢

热点阅读