简介

通俗的来说,Jackson是一个 Java 用来处理 JSON 格式数据的类库,其性能非常好。Jackson具有比较高的序列化和反序列化效率,据测试,无论是哪种形式的转换,Jackson > Gson > Json-lib,而且Jackson的处理能力甚至高出Json-lib近10倍左右,且正确性也十分高。

使用

Jackson提供了很多类和方法,而在序列化和反序列化中使用的最多的类则是ObjectMapper这个类,此类比较类似于Json-lib中JsonObject和ArrayObject。此类中提供了readTree(),readValue(),writeValueAsString()等方法用于转换。

1 常用注解

1、 @JsonProperty 此注解用于属性上,作用是把该属性的名称序列化为另外一个名称,如把testPwd属性序列化为pwd,@JsonProperty(value="pwd")。

2、@JsonPropertyOrder
作用在类上,被用来指明当序列化时需要对属性做排序,它有2个属性一个是alphabetic:布尔类型,表示是否采用字母拼音顺序排序,默认是为false,即不排序。如@JsonPropertyOrder(alphabetic=true)。

3、@JsonInclude
是用在实体类的方法类的头上 作用是实体类的参数查询到的为null的不显示,比如说你想传一些json数据到前台,但是不想传值为null的数据,就可以使用该标签。如@JsonInclude(JsonInclude.Include.NON_NULL)

4、@JsonIgnoreProperties
可以注明是想要忽略的属性列表如@JsonIgnoreProperties({"name","age","title"}),也可以注明过滤掉未知的属性如@JsonIgnoreProperties(ignoreUnknown=true),@JsonIgnore表示忽略当前属性。

5、@JsonFormat
用在属性和方法上,可以方便的进行格式转换,如把Date转换为我们要的模式@JsonFormat(pattern = "yyyy-MM-dd HH-mm-ss")。

6、@JsonUnwrapped
当实体类中成员属性是一个类的对象时候,忽略包装。直接显示属性。

2 序列化

/**
 * 测试用户类
 *
 * @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;
2.1 对自定义类的序列化


/**
   * @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());
    List<String> jsonList = new ArrayList<String>();
    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"}
["大锤","二锤"]


2.2 另一种序列化方法

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"}


3 反序列化

3.1 一次性反序列化

此方法中主要利用ObjectMapper提供的<T> readValue(String content, Class<T> valueType)方法。此方法需要输入Json串以及对应的需要填充的类的Class,返回填充后的类。将Json串解析到自定义类中。


**
 * @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<String, Map<String, Object>> 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更占内存一些。而利用空间换时间,一般是值得的。