https://github.com/zq2599/blog_demos

内容:所有原创文章分类汇总及配套源码,涉及Java、Docker、Kubernetes、DevOPS等;

这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos

  1. 本文主要内容是jackson-core库,这是个低阶API库,提供流式解析工具JsonParser,流式生成工具JsonGenerator
  2. 在日常的序列化和反序列化处理中,最常用的是jackson-annotationsjackson-databind,而jackson-core由于它提供的API过于基础,我们大多数情况下是用不上的;
  3. 尽管jackson-databind负责序列化和反序列化处理,但它的底层实现是调用了jackson-core的API;
  4. 本着万丈高楼平地起的原则,本文咱们通过实战了解神秘的jackson-core,了解整个jackson的序列化和反序列化基本原理;
  1. 如果您不想编码,可以在GitHub下载所有源码,地址和链接信息如下表所示(https://github.com/zq2599/blog_demos):
名称 链接 备注
项目主页 https://github.com/zq2599/blog_demos 该项目在GitHub上的主页
git仓库地址(https) https://github.com/zq2599/blog_demos.git 该项目源码的仓库地址,https协议
git仓库地址(ssh) git@github.com:zq2599/blog_demos.git 该项目源码的仓库地址,ssh协议
  1. 这个git项目中有多个文件夹,本章的应用在jacksondemo文件夹下,如下图红框所示:

在这里插入图片描述

创建名为jacksondemo的maven工程,这是个父子结构的工程,其pom.xml内容如下:

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
  4. <modelVersion>4.0.0</modelVersion>
  5. <properties>
  6. <java.version>1.8</java.version>
  7. </properties>
  8. <groupId>com.bolingcavalry</groupId>
  9. <artifactId>jacksondemo</artifactId>
  10. <version>1.0-SNAPSHOT</version>
  11. <packaging>pom</packaging>
  12. <modules>
  13. <module>core</module>
  14. <module>beans</module>
  15. <module>databind</module>
  16. </modules>
  17. <dependencyManagement>
  18. <dependencies>
  19. <dependency>
  20. <groupId>com.fasterxml.jackson.core</groupId>
  21. <artifactId>jackson-databind</artifactId>
  22. <version>2.11.0</version>
  23. <scope>compile</scope>
  24. </dependency>
  25. <dependency>
  26. <groupId>org.slf4j</groupId>
  27. <artifactId>slf4j-log4j12</artifactId>
  28. <version>1.7.25</version>
  29. <scope>compile</scope>
  30. </dependency>
  31. <dependency>
  32. <groupId>commons-io</groupId>
  33. <artifactId>commons-io</artifactId>
  34. <version>2.7</version>
  35. <scope>compile</scope>
  36. </dependency>
  37. <dependency>
  38. <groupId>org.apache.commons</groupId>
  39. <artifactId>commons-lang3</artifactId>
  40. <version>3.10</version>
  41. <scope>compile</scope>
  42. </dependency>
  43. </dependencies>
  44. </dependencyManagement>
  45. </project>
  1. 在父工程jscksondemo下新增名为beans的子工程,这里面是一些常量和Pojo类;
  2. 增加定义常量的类Constant.java:
  1. package com.bolingcavalry.jacksondemo.beans;
  2. public class Constant {
  3. /**
  4. * 该字符串的值是个网络地址,该地址对应的内容是个JSON
  5. */
  6. public final static String TEST_JSON_DATA_URL = "https://raw.githubusercontent.com/zq2599/blog_demos/master/files/twitteer_message.json";
  7. /**
  8. * 用来验证反序列化的JSON字符串
  9. */
  10. public final static String TEST_JSON_STR = "{\n" +
  11. " \"id\":1125687077,\n" +
  12. " \"text\":\"@stroughtonsmith You need to add a \\\"Favourites\\\" tab to TC/iPhone. Like what TwitterFon did. I can't WAIT for your Twitter App!! :) Any ETA?\",\n" +
  13. " \"fromUserId\":855523, \n" +
  14. " \"toUserId\":815309,\n" +
  15. " \"languageCode\":\"en\"\n" +
  16. "}";
  17. /**
  18. * 用来验证序列化的TwitterEntry实例
  19. */
  20. public final static TwitterEntry TEST_OBJECT = new TwitterEntry();
  21. /**
  22. * 准备好TEST_OBJECT对象的各个参数
  23. */
  24. static {
  25. TEST_OBJECT.setId(123456L);
  26. TEST_OBJECT.setFromUserId(101);
  27. TEST_OBJECT.setToUserId(102);
  28. TEST_OBJECT.setText("this is a message for serializer test");
  29. TEST_OBJECT.setLanguageCode("zh");
  30. }}
  1. 增加一个Pojo,对应的是一条推特消息:
  1. package com.bolingcavalry.jacksondemo.beans;
  2. /**
  3. * @Description: 推特消息bean
  4. * @author: willzhao E-mail: zq2599@gmail.com
  5. * @date: 2020/7/4 16:24
  6. */
  7. public class TwitterEntry {
  8. /**
  9. * 推特消息id
  10. */
  11. long id;
  12. /**
  13. * 消息内容
  14. */
  15. String text; /**
  16. * 消息创建者
  17. */
  18. int fromUserId;
  19. /**
  20. * 消息接收者
  21. */
  22. int toUserId;
  23. /**
  24. * 语言类型
  25. */
  26. String languageCode; public long getId() {
  27. return id;
  28. } public void setId(long id) {
  29. this.id = id;
  30. } public String getText() {
  31. return text;
  32. } public void setText(String text) {
  33. this.text = text;
  34. } public int getFromUserId() {
  35. return fromUserId;
  36. } public void setFromUserId(int fromUserId) {
  37. this.fromUserId = fromUserId;
  38. } public int getToUserId() {
  39. return toUserId;
  40. } public void setToUserId(int toUserId) {
  41. this.toUserId = toUserId;
  42. } public String getLanguageCode() {
  43. return languageCode;
  44. } public void setLanguageCode(String languageCode) {
  45. this.languageCode = languageCode;
  46. } public TwitterEntry() {
  47. } public String toString() {
  48. return "[Tweet, id: "+id+", text='"+text+"', from: "+fromUserId+", to: "+toUserId+", lang: "+languageCode+"]";
  49. }}
  1. 以上就是准备工作了,接下来开始实战jackson-core;
  1. JsonFactory是否是线程安全的,这是编码前要弄清楚的问题,因为JsonParserJsonGenerator的创建都离不开JsonFactory;
  2. 如下图红框所示,jackson官方文档中明确指出JsonFactory是线程安全的,可以放心的作为全局变量给多线程同时使用:

在这里插入图片描述
3. 官方文档地址:http://fasterxml.github.io/jackson-core/javadoc/2.11/

  1. 新建子工程core,pom.xml如下:
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  5. <parent>
  6. <artifactId>jacksondemo</artifactId>
  7. <groupId>com.bolingcavalry</groupId>
  8. <version>1.0-SNAPSHOT</version>
  9. <relativePath>../pom.xml</relativePath>
  10. </parent>
  11. <modelVersion>4.0.0</modelVersion>
  12. <groupId>com.bolingcavalry</groupId>
  13. <artifactId>core</artifactId>
  14. <name>core</name>
  15. <description>Demo project for jackson core use</description>
  16. <build>
  17. <plugins>
  18. <plugin>
  19. <groupId>org.apache.maven.plugins</groupId>
  20. <artifactId>maven-compiler-plugin</artifactId>
  21. <configuration>
  22. <source>8</source>
  23. <target>8</target>
  24. </configuration>
  25. </plugin>
  26. </plugins>
  27. </build>
  28. <dependencies>
  29. <dependency>
  30. <groupId>com.fasterxml.jackson.core</groupId>
  31. <artifactId>jackson-databind</artifactId>
  32. </dependency>
  33. <dependency>
  34. <groupId>org.slf4j</groupId>
  35. <artifactId>slf4j-log4j12</artifactId>
  36. </dependency>
  37. <dependency>
  38. <groupId>commons-io</groupId>
  39. <artifactId>commons-io</artifactId>
  40. </dependency>
  41. <dependency>
  42. <groupId>org.apache.commons</groupId>
  43. <artifactId>commons-lang3</artifactId>
  44. </dependency>
  45. <dependency>
  46. <groupId>com.bolingcavalry</groupId>
  47. <artifactId>beans</artifactId>
  48. <version>${project.version}</version>
  49. </dependency>
  50. </dependencies>
  51. </project>
  1. 新建StreamingDemo类,这里面是调用jackson-core的API进行序列化和反序列化的所有demo,如下:
  1. package com.bolingcavalry.jacksondemo.core;
  2. import com.bolingcavalry.jacksondemo.beans.TwitterEntry;
  3. import com.fasterxml.jackson.core.*;
  4. import com.fasterxml.jackson.databind.ObjectMapper;
  5. import org.apache.commons.io.IOUtils;
  6. import org.apache.commons.lang3.StringUtils;
  7. import org.slf4j.Logger;
  8. import org.slf4j.LoggerFactory;
  9. import java.io.ByteArrayOutputStream;
  10. import java.io.IOException;
  11. import java.net.URL;
  12. /**
  13. * @Description: jackson低阶方法的使用
  14. * @author: willzhao E-mail: zq2599@gmail.com
  15. * @date: 2020/7/4 15:50
  16. */
  17. public class StreamingDemo {
  18. private static final Logger logger = LoggerFactory.getLogger(StreamingDemo.class);
  19. JsonFactory jsonFactory = new JsonFactory();
  20. /**
  21. * 该字符串的值是个网络地址,该地址对应的内容是个JSON
  22. */
  23. final static String TEST_JSON_DATA_URL = "https://raw.githubusercontent.com/zq2599/blog_demos/master/files/twitteer_message.json";
  24. /**
  25. * 用来验证反序列化的JSON字符串
  26. */
  27. final static String TEST_JSON_STR = "{\n" +
  28. " \"id\":1125687077,\n" +
  29. " \"text\":\"@stroughtonsmith You need to add a \\\"Favourites\\\" tab to TC/iPhone. Like what TwitterFon did. I can't WAIT for your Twitter App!! :) Any ETA?\",\n" +
  30. " \"fromUserId\":855523, \n" +
  31. " \"toUserId\":815309,\n" +
  32. " \"languageCode\":\"en\"\n" +
  33. "}";
  34. /**
  35. * 用来验证序列化的TwitterEntry实例
  36. */
  37. final static TwitterEntry TEST_OBJECT = new TwitterEntry();
  38. /**
  39. * 准备好TEST_OBJECT对象的各个参数
  40. */
  41. static {
  42. TEST_OBJECT.setId(123456L);
  43. TEST_OBJECT.setFromUserId(101);
  44. TEST_OBJECT.setToUserId(102);
  45. TEST_OBJECT.setText("this is a message for serializer test");
  46. TEST_OBJECT.setLanguageCode("zh");
  47. }
  48. /**
  49. * 反序列化测试(JSON -> Object),入参是JSON字符串
  50. * @param json JSON字符串
  51. * @return
  52. * @throws IOException
  53. */
  54. public TwitterEntry deserializeJSONStr(String json) throws IOException {
  55. JsonParser jsonParser = jsonFactory.createParser(json);
  56. if (jsonParser.nextToken() != JsonToken.START_OBJECT) {
  57. jsonParser.close();
  58. logger.error("起始位置没有大括号");
  59. throw new IOException("起始位置没有大括号");
  60. }
  61. TwitterEntry result = new TwitterEntry();
  62. try {
  63. // Iterate over object fields:
  64. while (jsonParser.nextToken() != JsonToken.END_OBJECT) {
  65. String fieldName = jsonParser.getCurrentName();
  66. logger.info("正在解析字段 [{}]", jsonParser.getCurrentName());
  67. // 解析下一个
  68. jsonParser.nextToken();
  69. switch (fieldName) {
  70. case "id":
  71. result.setId(jsonParser.getLongValue());
  72. break;
  73. case "text":
  74. result.setText(jsonParser.getText());
  75. break;
  76. case "fromUserId":
  77. result.setFromUserId(jsonParser.getIntValue());
  78. break;
  79. case "toUserId":
  80. result.setToUserId(jsonParser.getIntValue());
  81. break;
  82. case "languageCode":
  83. result.setLanguageCode(jsonParser.getText());
  84. break;
  85. default:
  86. logger.error("未知字段 '" + fieldName + "'");
  87. throw new IOException("未知字段 '" + fieldName + "'");
  88. }
  89. }
  90. } catch (IOException e) {
  91. logger.error("反序列化出现异常 :", e);
  92. } finally {
  93. jsonParser.close(); // important to close both parser and underlying File reader
  94. }
  95. return result;
  96. }
  97. /**
  98. * 反序列化测试(JSON -> Object),入参是JSON字符串
  99. * @param url JSON字符串的网络地址
  100. * @return
  101. * @throws IOException
  102. */
  103. public TwitterEntry deserializeJSONFromUrl(String url) throws IOException {
  104. // 从网络上取得JSON字符串
  105. String json = IOUtils.toString(new URL(TEST_JSON_DATA_URL), JsonEncoding.UTF8.name());
  106. logger.info("从网络取得JSON数据 :\n{}", json);
  107. if(StringUtils.isNotBlank(json)) {
  108. return deserializeJSONStr(json);
  109. } else {
  110. logger.error("从网络获取JSON数据失败");
  111. return null;
  112. }
  113. }
  114. /**
  115. * 序列化测试(Object -> JSON)
  116. * @param twitterEntry
  117. * @return 由对象序列化得到的JSON字符串
  118. */
  119. public String serialize(TwitterEntry twitterEntry) throws IOException{
  120. String rlt = null;
  121. ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
  122. JsonGenerator jsonGenerator = jsonFactory.createGenerator(byteArrayOutputStream, JsonEncoding.UTF8);
  123. try {
  124. jsonGenerator.useDefaultPrettyPrinter();
  125. jsonGenerator.writeStartObject();
  126. jsonGenerator.writeNumberField("id", twitterEntry.getId());
  127. jsonGenerator.writeStringField("text", twitterEntry.getText());
  128. jsonGenerator.writeNumberField("fromUserId", twitterEntry.getFromUserId());
  129. jsonGenerator.writeNumberField("toUserId", twitterEntry.getToUserId());
  130. jsonGenerator.writeStringField("languageCode", twitterEntry.getLanguageCode());
  131. jsonGenerator.writeEndObject();
  132. } catch (IOException e) {
  133. logger.error("序列化出现异常 :", e);
  134. } finally {
  135. jsonGenerator.close();
  136. }
  137. // 一定要在
  138. rlt = byteArrayOutputStream.toString();
  139. return rlt;
  140. }
  141. public static void main(String[] args) throws Exception {
  142. StreamingDemo streamingDemo = new StreamingDemo();
  143. // 执行一次对象转JSON操作
  144. logger.info("********************执行一次对象转JSON操作********************");
  145. String serializeResult = streamingDemo.serialize(TEST_OBJECT);
  146. logger.info("序列化结果是JSON字符串 : \n{}\n\n", serializeResult);
  147. // 用本地字符串执行一次JSON转对象操作
  148. logger.info("********************执行一次本地JSON反序列化操作********************");
  149. TwitterEntry deserializeResult = streamingDemo.deserializeJSONStr(TEST_JSON_STR);
  150. logger.info("\n本地JSON反序列化结果是个java实例 : \n{}\n\n", deserializeResult);
  151. // 用网络地址执行一次JSON转对象操作
  152. logger.info("********************执行一次网络JSON反序列化操作********************");
  153. deserializeResult = streamingDemo.deserializeJSONFromUrl(TEST_JSON_DATA_URL);
  154. logger.info("\n网络JSON反序列化结果是个java实例 : \n{}", deserializeResult);
  155. ObjectMapper a;
  156. }
  157. }
  1. 上述代码可见JsonParser负责将JSON解析成对象的变量值,核心是循环处理JSON中的所有内容;
  2. JsonGenerator负责将对象的变量写入JSON的各个属性,这里是开发者自行决定要处理哪些字段;
  3. 不论是JsonParser还是JsonGenerator,大家都可以感觉到工作量很大,需要开发者自己动手实现对象和JSON字段的关系映射,实际应用中不需要咱们这样辛苦的编码,jackson的另外两个库(annonation的databind)已经帮我们完成了大量工作,上述代码只是揭示最基础的jackson执行原理;
  4. 执行StreamingDemo类,得到结果如下,序列化和反序列化都成功了:

在这里插入图片描述

  • 以上就是jackson-core的基本功能,咱们了解了jackson最底层的工作原理,接下来的文章会继续实践更多操作;
  1. Java系列
  2. Spring系列
  3. Docker系列
  4. kubernetes系列
  5. 数据库+中间件系列
  6. DevOps系列

微信搜索「程序员欣宸」,我是欣宸,期待与您一同畅游Java世界…
https://github.com/zq2599/blog_demos

版权声明:本文为bolingcavalry原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/bolingcavalry/p/14418971.html