通俗的来说,Jackson是一个 Java 用来处理 JSON 格式数据的类库,其性能非常好。Jackson具有比较高的序列化和反序列化效率,据测试,无论是哪种形式的转换,Jackson > Gson > Json-lib,而且Jackson的处理能力甚至高出Json-lib近10倍左右,且正确性也十分高。
Jackson提供了很多类和方法,而在序列化和反序列化中使用的最多的类则是ObjectMapper这个类,此类比较类似于Json-lib中JsonObject和ArrayObject。此类中提供了readTree(),readValue(),writeValueAsString()等方法用于转换。
1、 此注解用于属性上,作用是把该属性的名称序列化为另外一个名称,如把testPwd属性序列化为pwd, (value="pwd")。
2、
作用在类上,被用来指明当序列化时需要对属性做排序,它有2个属性一个是alphabetic:布尔类型,表示是否采用字母拼音顺序排序,默认是为false,即不排序。如
3、
是用在实体类的方法类的头上 作用是实体类的参数查询到的为null的不显示,比如说你想传一些json数据到前台,但是不想传值为null的数据,就可以使用该标签。如@JsonInclude(JsonInclude.Include.NON_NULL)
4、
可以注明是想要忽略的属性列表如 ({"name","age","title"}),也可以注明过滤掉未知的属性如 (ignoreUnknown=true),@JsonIgnore表示忽略当前属性。
5、
用在属性和方法上,可以方便的进行格式转换,如把Date转换为我们要的模式@JsonFormat(pattern = "yyyy-MM-dd HH-mm-ss")。
6、
当实体类中成员属性是一个类的对象时候,忽略包装。直接显示属性。
/** * 测试用户类 * * @author 杨小华 * @create 2017-09-27 9:39 */ @JsonInclude(JsonInclude.Include.NON_NULL) @JsonPropertyOrder(alphabetic=true) public class User{ /** * id * 不JSON序列化id */ @JsonIgnore private Integer id; /** * 昵称 */ private String name; /** * 密码 * 序列化testPwd属性为pwd */ @JsonProperty("pwd") private String testPwd; /** * 注册时间 * 格式化日期属性 */ @JsonFormat(pattern = "yyyy-MM-dd") private Date time;
/** * @desc JAVA对象和集合转JSON[JSON序列化] * @author 杨小华 * @create 2017/9/27 10:15 **/ @Test public void write() { User user = new User(); user.setId(1); user.setName("测试"); user.setTestPwd("123456"); user.setTime(new Date()); ListjsonList = new ArrayList (); jsonList.add("王大锤"); jsonList.add("王尼玛"); ObjectMapper objectMapper = new ObjectMapper(); try { String json = objectMapper.writeValueAsString(user); System.out.printf(json); String list = objectMapper.writeValueAsString(jsonList); System.out.printf(list); } catch (JsonProcessingException e) { e.printStackTrace(); } } 输出结果: {"name":"测试","time":"2017-09-27","pwd":"123456"} ["大锤","二锤"]
objectMapper.getFactory().createGenerator()此方法不仅可以将Json直接写入网络流,还可以将Json写入文件流或者内存流。所以用途更广。
/** * @desc 另一种序列化方法 * @author 杨小华 * @create 2017/9/27 10:30 **/ @Test public void jsonGenerator() { ObjectMapper objectMapper = new ObjectMapper(); User user = new User(); user.setId(1); user.setName("测试"); user.setTestPwd("123456"); user.setTime(new Date()); try { JsonGenerator jsonGenerator = objectMapper.getFactory().createGenerator(System.out, JsonEncoding.UTF8); jsonGenerator.writeObject(user); } catch (IOException e) { e.printStackTrace(); } } 输出结果:{"name":"测试","pwd":"123456","time":"2017-09-27"}
此方法中主要利用ObjectMapper提供的
** * @desc JSON转Java类[JSON反序列化] * @author 杨小华 * @create 2017/9/27 10:17 **/ @Test public void read() { String json = "{\"name\":\"测试\",\"time\":\"2017-09-27\",\"pwd\":\"123456\"}"; ObjectMapper objectMapper = new ObjectMapper(); try { User user = objectMapper.readValue(json, User.class); System.out.printf(user.getName()); } catch (IOException e) { e.printStackTrace(); } } 输出结果: 测试 /** * @desc JSON转Map类[JSON反序列化] * @author 杨小华 * @create 2017/9/27 10:43 **/ @Test public void jsonMap() { ObjectMapper objectMapper = new ObjectMapper(); String json = "{\"error\":0,\"data\":{\"name\":\"ABC\",\"age\":20,\"phone\":{\"home\":\"abc\",\"mobile\":\"def\"},\"friends\":[{\"name\":\"DEF\",\"phone\":{\"home\":\"hij\",\"mobile\":\"klm\"}}," + "{\"name\":\"GHI\",\"phone\":{\"home\":\"nop\",\"mobile\":\"qrs\"}}]},\"other\":{\"nickname\":[]}}"; try { Map> map = objectMapper.readValue(json, Map.class); System.out.printf(map.get("error") + "\n"); System.out.printf(map.get("data").get("phone") + ""); } catch (IOException e) { e.printStackTrace(); } } 输出结果: 0 {home=abc, mobile=def}
3.2 渐次反序列化
此方法更灵活,可以只将用户感兴趣的Json串信息值提取出来。主要利用ObjectMapper提供的readTree和Jackson提供的JsonNode类来实现(此方法类似于XML解析中的DOM方式解析,其好处是结构明细,便于提取想要的信息。当然,其缺点也一样:耗时费空间)。
/** * @desc 渐次反序列化 * @author 杨小华 * @create 2017/9/27 11:04 **/ @Test public void jsonTree() { ObjectMapper objectMapper = new ObjectMapper(); String test = "{\"results\":[{\"objectID\":357,\"geoPoints\":[{\"x\":504604.59802246094,\"y\":305569.9150390625}]},{\"objectID\":358,\"geoPoints\":[{\"x\":504602.2680053711,\"y\":305554.43603515625}]}]}"; try { JsonNode jsonNode = objectMapper.readTree(test);//将Json串以树状结构读入内存 JsonNode contents = jsonNode.get("results");//得到results这个节点下的信息 for (int i = 0; i < contents.size(); i++) { System.out.printf(contents.get(i).get("objectID").intValue() + "\n"); JsonNode geoContent = contents.get(i).get("geoPoints");//得到geoPoints这个节点下的信息 for (int j = 0; j < geoContent.size(); j++) { System.out.printf(geoContent.get(j).get("x").doubleValue() + "\n"); System.out.printf(geoContent.get(j).get("y").decimalValue() + "\n"); } } } catch (IOException e) { e.printStackTrace(); } }
Jackson关于Json的操作主要如上所示,其方法使用起来很便利,而且也很灵活,即提供了一次性完成的操作,也提供了可以按需读取信息的操作。并且Jackson的功能很齐全,可以对序列化和反序列化进行多种细节的控制,例如注解功能和对于Hibernate的延迟注入功能以及设置时间格式功能等,因为这些功能目前不太需要,所以仔细研究留待以后。同时,Jackson还支持对XML的一系列序列化和反序列化的操作,其思路与解析Json的大致相同。
对于Jackson目前的缺点,网上有人测试所比Json-lib更占内存一些。而利用空间换时间,一般是值得的。