前言
因为自己维护着一款课程表软件,软件里有通过二维码分享课程的功能,大致实现就是将课程信息转换为 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算法与哈夫曼编码的一个无损数据压缩算法。
我们可以使用 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
二维码对比
可以看到二维码的内容已经少了很多,这次的优化就到此为止了。
欢迎留言探讨,谢谢大家。