JSON 解析 (三)—— FastJSON与Jackson比较
一、方便性与性能
调用方便性而言:
- FastJSON提供了大量静态方法,调用简洁方便
- Jackson须实例化类,调用相对繁琐,可通过封装成JSON工具类简化调用
性能而言:
- FastJSON反序列化的性能略差,对于256k的json字符串,平均700ms
- Jackson 的 data binding反序列化的性能稍好,对于256k的json字符串,平均600ms
- 两者的序列化性能基本相同,对于256k的json字符串,平均140ms
- 相对data binding方式(ObjectMapper.writeValueAsString()),Jackson的流输出方式(JsonGenerator.writeObject())性能稍好,平均130ms
二、其他
1、序列化与反序列化的实现细节
对象在序列化时,通过反射,遍历getter方法(getXXX(),isXXX),从方法名中获取属性名(XXX),并调用getter方法获取属性值,组合在一起,从而完成序列化;在反序列化时,遍历属性名(XXX)并拼接成setter方法(setXXX), 通过反射,查询并调用setter方法,从而完成对象构建。 序列化与反序列化的实现,与具体字段无直接关联,示例如下:
public class Product { private String name; private Double price; // private boolean sold; public boolean isSold() { return true; } // name、price的getter、setter方法 ... }
@Test public void testBoolPOJO() { Product product = new Product(); product.setName("hello"); product.setPrice(12.36); String json = JSON.toJSONString(product); System.out.println(json); // {"name":"hello","price":12.36,"sold":true} Product product2 = JSON.parseObject(json, Product.class); System.out.println(product2.getName() + " " + product2.getPrice()); // hello 12.36 }
如上所示,sold字段不存在,但在序列化时,可依据getter方法,在结果中生成该字段;在反序列化时,依据setter方法,上例即便存在sold字段(去掉该字段注释),由于不存在该字段的setter方法,无法从json字符串中获取到该属性值。
《阿里巴巴Java编程规约》指出,在POJO类中,对布尔类型的变量,都不要加 is 前缀,如:定义为基本数据类型 Boolean isDeleted 的属性,它的方法是 isDeleted()、setIsDeleted(),在反向解析的时候,反序列化工具“误以为”对应的属性名称是 deleted,尝试调用setDeleted(),由于方法不存在,获取不到属性,可能抛出异常(如Jackson)
2、FastJSON与Jackson在反序列化时的细微差异
a、在反序列化时,默认情况下,如果根据Json字符串中的属性,在实体类中找不到对应的setter方法,FastJSON会忽略该属性,而Jackson会报错
测试Product实体如上
@Test public void testBoolPOJO2() throws Exception { Product product = new Product(); product.setName("hello"); product.setPrice(12.36); ObjectMapper objectMapper = new ObjectMapper(); // objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); String json = objectMapper.writeValueAsString(product); System.out.println(json); Product product2 = objectMapper.readValue(json, Product.class); // 报错 System.out.println(product2.getName() + " " + product2.getPrice()); }
通过配置DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES为fasle,可忽略多出的属性
spring mvc使用的MappingJackson2HttpMessageConverter,在设置ObjectMapper的DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES时,在spring 3中为true,在spring 4及之后为false,因此,请求的json实体可包含冗余字段。
详细可参考:关于jackson的DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES
b、反序列化时,若找不到属性setter方面,Jackson会进一步查找同名字段,FastJSON则直接忽略
public class Product { private String name; private Double price; private boolean sold; public boolean isSold() { return this.sold; } // name、price的getter、setter方法 ... }
@Test public void testBoolPOJO3() throws Exception { String json = "{\"name\":\"hello\",\"price\":12.36,\"sold\":true}"; ObjectMapper objectMapper = new ObjectMapper(); Product product1 = objectMapper.readValue(json, Product.class); System.out.println(product1.isSold()); // true Product product2 = JSON.parseObject(json, Product.class); System.out.println(product2.isSold()); // false }
参考:
关于jackson的DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES