文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Jackson之 Java JSON 解析器

2024-12-03 16:26

关注

在当今的编程世界里,JSON 已经成为将信息从客户端传输到服务器端的首选协议,可以好不夸张的说,XML 就是那个被拍死在沙滩上的前浪。

很不幸的是,JDK 没有 JSON 库,不知道为什么不搞一下。Log4j 的时候,为了竞争,还推出了 java.util.logging,虽然最后也没多少人用。

Java 之所以牛逼,很大的功劳在于它的生态非常完备,JDK 没有 JSON 库,第三方类库有啊,还挺不错,比如说本篇的猪脚——Jackson,GitHub 上标星 6.1k,Spring Boot 的默认 JSON 解析器。

怎么证明这一点呢?

当我们通过 starter 新建一个 Spring Boot 的 Web 项目后,就可以在 Maven 的依赖项中看到 Jackson 的身影。

Jackson 有很多优点:

Jackson 的核心模块由三部分组成:

01、引入 Jackson 依赖

要想使用 Jackson,需要在 pom.xml 文件中添加 Jackson 的依赖。

  1.  
  2.     com.fasterxml.jackson.core 
  3.     jackson-databind 
  4.     2.10.1 
  5.  

jackson-databind 依赖于 jackson-core 和 jackson-annotations,所以添加完 jackson-databind 之后,Maven 会自动将 jackson-core 和 jackson-annotations 引入到项目当中。

Maven 之所以讨人喜欢的一点就在这,能偷偷摸摸地帮我们把该做的做了。

02、使用 ObjectMapper

Jackson 最常用的 API 就是基于”对象绑定” 的 ObjectMapper,它通过 writeValue 的系列方法将 Java 对象序列化为 JSON,并且可以存储成不同的格式。

来看一下存储成字符串的代码示例:

  1. import com.fasterxml.jackson.core.JsonProcessingException; 
  2. import com.fasterxml.jackson.databind.ObjectMapper; 
  3.  
  4.  
  5. public class Demo { 
  6.     public static void main(String[] args) throws JsonProcessingException { 
  7.         Writer wanger = new Writer("沉默王二", 18); 
  8.         ObjectMapper mapper = new ObjectMapper(); 
  9.         String jsonString = mapper.writerWithDefaultPrettyPrinter() 
  10.                 .writeValueAsString(wanger); 
  11.         System.out.println(jsonString); 
  12.     } 
  13.  
  14. class Writer { 
  15.     private String name
  16.     private int age; 
  17.  
  18.     public Writer(String nameint age) { 
  19.         this.name = name
  20.         this.age = age; 
  21.     } 
  22.  
  23.     public String getName() { 
  24.         return name
  25.     } 
  26.  
  27.     public void setName(String name) { 
  28.         this.name = name
  29.     } 
  30.  
  31.     public int getAge() { 
  32.         return age; 
  33.     } 
  34.  
  35.     public void setAge(int age) { 
  36.         this.age = age; 
  37.     } 

程序输出结果如下所示:

  1.   "name" : "沉默王二"
  2.   "age" : 18 

不是所有的字段都支持序列化和反序列化,需要符合以下规则:

如果想更改默认的序列化和反序列化规则,需要调用 ObjectMapper 的setVisibility() 方法。否则将会抛出 InvalidDefinitionException 异常。

ObjectMapper 通过 readValue 的系列方法从不同的数据源将 JSON 反序列化为 Java 对象。

来看一下将字符串反序列化为 Java 对象的代码示例:

  1. import com.fasterxml.jackson.core.JsonProcessingException; 
  2. import com.fasterxml.jackson.databind.ObjectMapper; 
  3.  
  4. public class Demo { 
  5.     public static void main(String[] args) throws JsonProcessingException { 
  6.         ObjectMapper mapper = new ObjectMapper(); 
  7.         String jsonString = "{\n" + 
  8.                 "  \"name\" : \"沉默王二\",\n" + 
  9.                 "  \"age\" : 18\n" + 
  10.                 "}"
  11.         Writer deserializedWriter = mapper.readValue(jsonString, Writer.class); 
  12.         System.out.println(deserializedWriter); 
  13.     } 
  14.  
  15. class Writer{ 
  16.     private String name
  17.     private int age; 
  18.  
  19.     // getter/setter 
  20.  
  21.     @Override 
  22.     public String toString() { 
  23.         return "Writer{" + 
  24.                 "name='" + name + '\'' + 
  25.                 ", age=" + age + 
  26.                 '}'
  27.     } 

程序输出结果如下所示:

  1. Writer{name='沉默王二', age=18} 

PS:如果反序列化的对象有带参的构造方法,它必须有一个空的默认构造方法,否则将会抛出 InvalidDefinitionException 一行。

  1. Exception in thread "main" com.fasterxml.jackson.databind.exc.InvalidDefinitionException: Cannot construct instance of `com.itwanger.jackson.Writer` (no Creators, like default construct, exist): cannot deserialize from Object value (no delegate- or property-based Creator) 
  2.  at [Source: (String)"{ 
  3.   "name" : "沉默王二"
  4.   "age" : 18 
  5. }"; line: 2, column: 3] 
  6.  at com.fasterxml.jackson.databind.exc.InvalidDefinitionException.from(InvalidDefinitionException.java:67) 
  7.  at com.fasterxml.jackson.databind.DeserializationContext.reportBadDefinition(DeserializationContext.java:1589) 
  8.  at com.fasterxml.jackson.databind.DeserializationContext.handleMissingInstantiator(DeserializationContext.java:1055) 
  9.  at com.fasterxml.jackson.databind.deser.BeanDeserializerBase.deserializeFromObjectUsingNonDefault(BeanDeserializerBase.java:1297) 
  10.  at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserializeFromObject(BeanDeserializer.java:326) 
  11.  at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:159) 
  12.  at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:4202) 
  13.  at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:3205) 
  14.  at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:3173) 
  15.  at com.itwanger.jackson.Demo.main(Demo.java:19) 

Jackson 最常用的 API 就是基于”对象绑定” 的 ObjectMapper,

ObjectMapper 也可以将 JSON 解析为基于“树模型”的 JsonNode 对象,来看下面的示例。

  1. import com.fasterxml.jackson.core.JsonProcessingException; 
  2. import com.fasterxml.jackson.databind.JsonNode; 
  3. import com.fasterxml.jackson.databind.ObjectMapper; 
  4.  
  5. public class JsonNodeDemo { 
  6.     public static void main(String[] args) throws JsonProcessingException { 
  7.         ObjectMapper mapper = new ObjectMapper(); 
  8.         String json = "{ \"name\" : \"沉默王二\", \"age\" : 18 }"
  9.         JsonNode jsonNode = mapper.readTree(json); 
  10.         String name = jsonNode.get("name").asText(); 
  11.         System.out.println(name); // 沉默王二 
  12.     } 

借助 TypeReference 可以将 JSON 字符串数组转成泛型 List,来看下面的示例:

  1. import com.fasterxml.jackson.core.JsonProcessingException; 
  2. import com.fasterxml.jackson.core.type.TypeReference; 
  3. import com.fasterxml.jackson.databind.ObjectMapper; 
  4.  
  5. import java.util.List; 
  6.  
  7. public class TypeReferenceDemo { 
  8.     public static void main(String[] args) throws JsonProcessingException { 
  9.         ObjectMapper mapper = new ObjectMapper(); 
  10.         String json = "[{ \"name\" : \"沉默王三\", \"age\" : 18 }, { \"name\" : \"沉默王二\", \"age\" : 19 }]"
  11.         List listAuthor = mapper.readValue(json, new TypeReference>(){}); 
  12.         System.out.println(listAuthor); 
  13.     } 
  14. class Author{ 
  15.     private String name
  16.     private int age; 
  17.  
  18.     // getter/setter 
  19.  
  20.     // toString 

03、更高级的配置

Jackson 之所以牛掰的一个很重要的因素是可以实现高度灵活的自定义配置。

在实际的应用场景中,JSON 中常常会有一些 Java 对象中没有的字段,这时候,如果直接解析的话,会抛出 UnrecognizedPropertyException 异常。

下面是一串 JSON 字符串:

  1. String jsonString = "{\n" + 
  2.                 "  \"name\" : \"沉默王二\",\n" + 
  3.                 "  \"age\" : 18\n" + 
  4.                 "  \"sex\" : \"男\",\n" + 
  5.                 "}"

但 Java 对象 Writer 中没有定义 sex 字段:

  1. class Writer{ 
  2.     private String name
  3.     private int age; 
  4.  
  5.     // getter/setter 

我们来尝试解析一下:

  1. ObjectMapper mapper = new ObjectMapper(); 
  2. Writer deserializedWriter = mapper.readValue(jsonString, Writer.class); 

不出意外,抛出异常了,sex 无法识别。

  1. Exception in thread "main" com.fasterxml.jackson.databind.exc.UnrecognizedPropertyException: Unrecognized field "sex" (class com.itwanger.jackson.Writer), not marked as ignorable (2 known properties: "name""age"]) 
  2.  at [Source: (String)"{ 
  3.   "name" : "沉默王二"
  4.   "age" : 18, 
  5.   "sex" : "男" 
  6. }"; line: 4, column: 12] (through reference chain: com.itwanger.jackson.Writer["sex"]) 

怎么办呢?可以通过 configure() 方法忽略掉这些“无法识别”的字段。

  1. mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); 

除此之外,还有其他一些有用的配置信息,来了解一下:

  1. // 在序列化时忽略值为 null 的属性 
  2. mapper.setSerializationInclusion(JsonInclude.Include.NON_NULL); 
  3. // 忽略值为默认值的属性 
  4. mapper.setDefaultPropertyInclusion(JsonInclude.Include.NON_DEFAULT); 

04、处理日期格式

对于日期类型的字段,比如说 java.util.Date,如果不指定格式,序列化后将显示为 long 类型的数据,这种默认格式的可读性很差。

  1.   "age" : 18, 
  2.   "birthday" : 1606358621209 

怎么办呢?

第一种方案,在 getter 上使用 @JsonFormat 注解。

  1. private Date birthday; 
  2.  
  3. // GMT+8 是指格林尼治的标准时间,在加上八个小时表示你现在所在时区的时间 
  4. @JsonFormat(timezone = "GMT+8", pattern = "yyyy-MM-dd HH:mm:ss"
  5. public Date getBirthday() { 
  6.     return birthday; 
  7.  
  8. public void setBirthday(Date birthday) { 
  9.     this.birthday = birthday; 

再来看一下结果:

  1.   "age" : 18, 
  2.   "birthday" : "2020-11-26 03:02:30" 

具体代码如下所示:

  1. ObjectMapper mapper = new ObjectMapper(); 
  2. Writer wanger = new Writer("沉默王二", 18); 
  3. wanger.setBirthday(new Date()); 
  4. String jsonString = mapper.writerWithDefaultPrettyPrinter() 
  5.                 .writeValueAsString(wanger); 
  6. System.out.println(jsonString); 

第二种方案,调用 ObjectMapper 的 setDateFormat() 方法。

  1. ObjectMapper mapper = new ObjectMapper(); 
  2. mapper.setDateFormat(StdDateFormat.getDateTimeInstance()); 
  3. Writer wanger = new Writer("沉默王二", 18); 
  4. wanger.setBirthday(new Date()); 
  5. String jsonString = mapper.writerWithDefaultPrettyPrinter() 
  6.                 .writeValueAsString(wanger); 
  7. System.out.println(jsonString); 

输出结果如下所示:

  1.   "name" : "沉默王二"
  2.   "age" : 18, 
  3.   "birthday" : "2020年11月26日 上午11:09:51" 

05、字段过滤

在将 Java 对象序列化为 JSON 时,可能有些字段需要过滤,不显示在 JSON 中,Jackson 有一种比较简单的实现方式。

@JsonIgnore 用于过滤单个字段。

  1. @JsonIgnore 
  2. public String getName() { 
  3.     return name

@JsonIgnoreProperties 用于过滤多个字段。

  1. @JsonIgnoreProperties(value = { "age","birthday" }) 
  2. class Writer{ 
  3.     private String name
  4.     private int age; 
  5.     private Date birthday; 

06、自定义序列化和反序列化

当 Jackson 默认序列化和反序列化不能满足实际的开发需要时,可以自定义新的序列化和反序列化类。

自定义的序列化类需要继承 StdSerializer,同时重写 serialize() 方法,利用 JsonGenerator 生成 JSON,示例如下:

  1. public class CustomSerializer extends StdSerializer { 
  2.     protected CustomSerializer(Class t) { 
  3.         super(t); 
  4.     } 
  5.  
  6.     public CustomSerializer() { 
  7.         this(null); 
  8.     } 
  9.  
  10.     @Override 
  11.     public void serialize(Man value, JsonGenerator gen, SerializerProvider provider) throws IOException { 
  12.         gen.writeStartObject(); 
  13.         gen.writeStringField("name", value.getName()); 
  14.         gen.writeEndObject(); 
  15.     } 
  16.  
  17. class Man{ 
  18.     private int age; 
  19.     private String name
  20.  
  21.     public Man(int age, String name) { 
  22.         this.age = age; 
  23.         this.name = name
  24.     } 
  25.  
  26.     public int getAge() { 
  27.         return age; 
  28.     } 
  29.  
  30.     public void setAge(int age) { 
  31.         this.age = age; 
  32.     } 
  33.  
  34.     public String getName() { 
  35.         return name
  36.     } 
  37.  
  38.     public void setName(String name) { 
  39.         this.name = name
  40.     } 

定义好自定义序列化类后,要想在程序中调用它们,需要将其注册到 ObjectMapper 的 Module 中,示例如下所示:

  1. ObjectMapper mapper = new ObjectMapper(); 
  2. SimpleModule module = 
  3.         new SimpleModule("CustomSerializer", new Version(1, 0, 0, nullnullnull)); 
  4. module.addSerializer(Man.class, new CustomSerializer()); 
  5. mapper.registerModule(module); 
  6. Man man = new Man( 18,"沉默王二"); 
  7. String json = mapper.writeValueAsString(man); 
  8. System.out.println(json); 

程序输出结果如下所示:

  1. {"name":"沉默王二"

自定义序列化类 CustomSerializer 中没有添加 age 字段,所以只输出了 name 字段。

再来看一下自定义的反序列化类,继承 StdDeserializer,同时重写 deserialize() 方法,利用 JsonGenerator 读取 JSON,示例如下:

  1. public class CustomDeserializer extends StdDeserializer { 
  2.     protected CustomDeserializer(Class vc) { 
  3.         super(vc); 
  4.     } 
  5.  
  6.     public CustomDeserializer() { 
  7.         this(null); 
  8.     } 
  9.  
  10.     @Override 
  11.     public Woman deserialize(JsonParser p, DeserializationContext ctxt) throws IOException, JsonProcessingException { 
  12.         JsonNode node = p.getCodec().readTree(p); 
  13.         Woman woman = new Woman(); 
  14.         int age = (Integer) ((IntNode) node.get("age")).numberValue(); 
  15.         String name = node.get("name").asText(); 
  16.         woman.setAge(age); 
  17.         woman.setName(name); 
  18.         return woman; 
  19.     } 
  20. class Woman{ 
  21.     private int age; 
  22.     private String name
  23.  
  24.     public Woman() { 
  25.     } 
  26.  
  27.     // getter/setter 
  28.  
  29.     @Override 
  30.     public String toString() { 
  31.         return "Woman{" + 
  32.                 "age=" + age + 
  33.                 ", name='" + name + '\'' + 
  34.                 '}'
  35.     } 

通过 JsonNode 把 JSON 读取到一个树形结构中,然后通过 JsonNode 的 get 方法将对应字段读取出来,然后生成新的 Java 对象,并返回。

定义好自定义反序列化类后,要想在程序中调用它们,同样需要将其注册到 ObjectMapper 的 Module 中,示例如下所示:

  1. ObjectMapper mapper = new ObjectMapper(); 
  2. SimpleModule module = 
  3.         new SimpleModule("CustomDeserializer", new Version(1, 0, 0, nullnullnull)); 
  4. module.addDeserializer(Woman.class, new CustomDeserializer()); 
  5. mapper.registerModule(module); 
  6. String json = "{ \"name\" : \"三妹\", \"age\" : 18 }"
  7. Woman woman = mapper.readValue(json, Woman.class); 
  8. System.out.println(woman); 

程序输出结果如下所示:

  1. Woman{age=18, name='三妹'

07、结语

哎呀,好像不错哦,Jackson 绝对配得上“最牛掰”这三个字。如果只想简单的序列化和反序列化,使用 ObjectMapper 的 write 和 read 方法即可。

如果还想更进一步的话,就需要对 ObjectMapper 进行一些自定义配置,或者加一些注解,以及直接自定义序列化和反序列化类,更贴近一些 Java 对象。

需要注意的是,对日期格式的字段要多加小心,尽量不要使用默认配置,可读性很差。

好了,通过这篇文章的系统化介绍,相信读者朋友们已经完全摸透 Jackson 了,我们下篇文章见。

本文转载自微信公众号「沉默王二」,可以通过以下二维码关注。转载本文请联系沉默王二公众号。

 

来源:沉默王二内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯