JSON转对象在开发中是一个常规且普遍的操作,为了增加开发效率经常会使用第三方JSON处理类库,其中Fastjson是比较受欢迎的JSON处理类库之一。
在最近的一次使用中,就遇到了一个问题,JSON转Map后因顺序发生了变化,造成依赖参数顺序的操作不能正常执行。
解决该问题的方法如下:
String jsonStr = "{\"J2\":\"A\",\"J1\":\"B\",\"listMap\":[{\"I2\":\"C\",\"I1\":\"D\",\"I3\":\"E\"}]}";
Map jsonMap = JSON.parseObject(jsonStr, Map.class, Feature.OrderedField);
在此之前的开发中,JSON转实体对象时,不需要关心转换后对象的顺序问题,所以并未深究顺序变化的原因。既然遇到了这个问题就查下原因吧。
{"J2":"A","J1":"B","listMap":[{"I2":"C","I1":"D","I3":"E"}]}
如以上JSON字符串,如果使用如下方式转为Map,就会出现转后的Map会按照KEY的自然顺序排序。
String jsonStr = "{\"J2\":\"A\",\"J1\":\"B\",\"listMap\":[{\"I2\":\"C\",\"I1\":\"D\",\"I3\":\"E\"}]}";
Map jsonMap = JSON.parseObject(jsonStr, Map.class);
转换后Map的排列顺序如下图:
出现这种问题,首先想到的是,Fastjson在转换Map时,实例化的Map为HashMap,HashMap本身有默认的排序算法,根据hashcode值进行排序。所以讲转换方式改为如下代码
String jsonStr = "{\"J2\":\"A\",\"J1\":\"B\",\"listMap\":[{\"I2\":\"C\",\"I1\":\"D\",\"I3\":\"E\"}]}";
Map jsonMap = JSON.parseObject(jsonStr, LinkedHasMap.class);
最终运行结果如下,只有外层Key顺序一致,内层嵌套集合转换时,顺序依然有变化。
图:
最终跟踪代码发现,Fastjson的DefaultJSONParser类下在转换内层Json依然会实例化一个Map对象,如下代码:
public JSONObject(int initialCapacity, boolean ordered) {
if (ordered) {
this.map = new LinkedHashMap(initialCapacity);
} else {
this.map = new HashMap(initialCapacity);
}
}
ordered默认为false,会默认实例化HashMap,所以内存嵌套依然是非固定顺序的。
所以只需要改变为ordered=true即可使用LinkedHashMap实例化Map对象。最终找到了解决了该问题的方法。
之前用Fastjson并未深入探究更多使用功能,现在发现还有Feature这个枚举对象,用于扩展Fastjson功能。具体Feature使用细则,后续再研究吧。