淺談JackSon的幾種用法
JackSon介紹
本文使用的JackSon版本為2.9.6。
JackSon是解析JSON和XML的一個框架,優(yōu)點是簡單易用,性能較高。
JackSon處理JSON的方式
JackSon提供了三種JSON的處理方式。分別是數(shù)據(jù)綁定,樹模型,流式API。下面會分別介紹這三種方式。
JackSon數(shù)據(jù)綁定
數(shù)據(jù)綁定用于JSON轉(zhuǎn)化,可以將JSON與POJO對象進行轉(zhuǎn)化。數(shù)據(jù)綁定有兩種,簡單數(shù)據(jù)綁定和完整數(shù)據(jù)綁定。
完整數(shù)據(jù)綁定
package com.xymxyg.json;
import com.fasterxml.jackson.databind.ObjectMapper;
import java.io.IOException;
/**
* @author guangsheng.tang
* 下面是最常用的場景,將json字符串映射為對象,或者是將對象轉(zhuǎn)化為json字符串。這是完整數(shù)據(jù)綁定。
缺點:這種方法十分方便,但是擴展性不強,增加一個字段便要修改POJO對象,這個操作有一定風(fēng)險性。并且解析的時候,如果json缺少POJO中的某字段,映射出的對象對應(yīng)值默認為null,直接使用有一定風(fēng)險。如果json對象多了某一字段,解析過程中會拋出UnrecognizedPropertyException異常。并且如果json較為復(fù)雜的話,POJO對象會顯得特別臃腫。
*/
public class CompleteDataBind {
public static void main(String[] args) throws IOException {
String s = "{\"id\": 1,\"name\": \"小明\",\"array\": [\"1\", \"2\"]}";
ObjectMapper mapper = new ObjectMapper();
//Json映射為對象
Student student = mapper.readValue(s, Student.class);
//對象轉(zhuǎn)化為Json
String json = mapper.writeValueAsString(student);
System.out.println(json);
System.out.println(student.toString());
}
}
package com.xymxyg.json;
/**
* @author guangsheng.tang
*/
public class Student {
private int id;
private String name;
private String sex;
private ArrayList<String> array;
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
public ArrayList<String> getArray() {
return array;
}
public void setArray(ArrayList<String> array) {
this.array = array;
}
@Override
public String toString() {
return "Student{" +
"id=" + id +
", name='" + name + '\'' +
", sex='" + sex + '\'' +
", array=" + Arrays.toString(array.toArray()) +
'}';
}
}
簡單數(shù)據(jù)綁定
簡單數(shù)據(jù)綁定就是將json字符串映射為java核心的數(shù)據(jù)類型。
| json類型 | Java類型 |
|---|---|
| object | LinkedHashMap |
| array | ArrayList |
| string | String |
| number | Integer,Long,Double |
| true|false | Boolean |
| null | null |
下面演示一個例子,將json轉(zhuǎn)化為一個Map。通過Map來讀取。
package com.xymxyg.json;
import com.fasterxml.jackson.databind.ObjectMapper;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
/**
* @author guangsheng.tang
* 簡單數(shù)據(jù)綁定的示例,不用POJO對象,直接映射為一個Map,然后從Map中獲取。
*/
public class SimpleDataBind {
public static void main(String[] args) throws IOException {
Map<String, Object> map = new HashMap<>(16);
String s = "{\"id\": 1,\"name\": \"小明\",\"array\": [\"1\", \"2\"]," +
"\"test\":\"I'm test\",\"base\": {\"major\": \"物聯(lián)網(wǎng)\",\"class\": \"3\"}}";
ObjectMapper mapper = new ObjectMapper();
map = mapper.readValue(s, map.getClass());
//獲取id
Integer studentId = (Integer) map.get("id");
System.out.println(studentId);
//獲取數(shù)據(jù)
ArrayList list = (ArrayList) map.get("array");
System.out.println(Arrays.toString(list.toArray()));
//新增加的字段可以很方便的處理
String test = (String) map.get("test");
System.out.println(test);
//不存在的返回null
String notExist = (String) map.get("notExist");
System.out.println(notExist);
//嵌套的對象獲取
Map base = (Map) map.get("base");
String major = (String) base.get("major");
System.out.println(major);
}
}
樹模型
針對JackSon的樹模型結(jié)構(gòu),我下面寫了一個比較完善的例子。同樣Java樹模型有優(yōu)點,也有缺點。
package com.xymxyg.json;
import com.fasterxml.jackson.core.JsonPointer;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.node.ArrayNode;
import com.fasterxml.jackson.databind.node.NullNode;
import com.fasterxml.jackson.databind.node.ObjectNode;
import java.io.IOException;
/**
* @author guangsheng.tang
* JackSon樹模型結(jié)構(gòu),可以通過get,JsonPointer等進行操作,適合用來獲取大Json中的字段,比較靈活。缺點是如果需要獲取的內(nèi)容較多,
* 會顯得比較繁瑣。
*/
public class TreeModel {
public static void main(String[] args) throws IOException {
ObjectMapper mapper = new ObjectMapper();
//以下是對象轉(zhuǎn)化為Json
JsonNode root = mapper.createObjectNode();
((ObjectNode) root).putArray("array");
ArrayNode arrayNode = (ArrayNode) root.get("array");
((ArrayNode) arrayNode).add("args1");
((ArrayNode) arrayNode).add("args2");
((ObjectNode) root).put("name", "小紅");
String json = mapper.writeValueAsString(root);
System.out.println("使用樹型模型構(gòu)建的json:"+json);
//以下是樹模型的解析Json
String s = "{\"id\": 1,\"name\": \"小明\",\"array\": [\"1\", \"2\"]," +
"\"test\":\"I'm test\",\"nullNode\":null,\"base\": {\"major\": \"物聯(lián)網(wǎng)\",\"class\": \"3\"}}";
//讀取rootNode
JsonNode rootNode = mapper.readTree(s);
//通過path獲取
System.out.println("通過path獲取值:" + rootNode.path("name").asText());
//通過JsonPointer可以直接按照路徑獲取
JsonPointer pointer = JsonPointer.valueOf("/base/major");
JsonNode node = rootNode.at(pointer);
System.out.println("通過at獲取值:" + node.asText());
//通過get可以取對應(yīng)的value
JsonNode classNode = rootNode.get("base");
System.out.println("通過get獲取值:" + classNode.get("major").asText());
//獲取數(shù)組的值
System.out.print("獲取數(shù)組的值:");
JsonNode arrayNode2 = rootNode.get("array");
for (int i = 0; i < arrayNode2.size(); i++) {
System.out.print(arrayNode2.get(i).asText()+" ");
}
System.out.println();
//path和get方法看起來很相似,其實他們的細節(jié)不同,get方法取不存在的值的時候,會返回null。而path方法會
//返回一個"missing node",該"missing node"的isMissingNode方法返回值為true,如果調(diào)用該node的asText方法的話,
// 結(jié)果是一個空字符串。
System.out.println("get方法取不存在的節(jié)點,返回null:" + (rootNode.get("notExist") == null));
JsonNode notExistNode = rootNode.path("notExist");
System.out.println("notExistNode的value:" + notExistNode.asText());
System.out.println("isMissingNode方法返回true:" + notExistNode.isMissingNode());
//當(dāng)key存在,而value為null的時候,get和path都會返回一個NullNode節(jié)點。
System.out.println(rootNode.get("nullNode") instanceof NullNode);
System.out.println(rootNode.path("nullNode") instanceof NullNode);
}
}
流式API
流式API是一套比較底層的API,速度快,但是使用起來特別麻煩。它主要是有兩個核心類,一個是JsonGenerator,用來生成json,另一個是JsonParser,用來讀取json內(nèi)容。話不多說,直接上代碼演示。
package com.xymxyg.json;
import com.fasterxml.jackson.core.*;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
/**
* @author guangsheng.tang
* JsonParser和Generator的優(yōu)點是速度快,缺點是寫起來真的很復(fù)雜。
*/
public class StreamApi {
public static void main(String[] args) throws IOException {
JsonFactory factory = new JsonFactory();
String s = "{\"id\": 1,\"name\": \"小明\",\"array\": [\"1\", \"2\"]," +
"\"test\":\"I'm test\",\"nullNode\":null,\"base\": {\"major\": \"物聯(lián)網(wǎng)\",\"class\": \"3\"}}";
//這里就舉一個比較簡單的例子,Generator的用法就是一個一個write即可。
File file = new File("/json.txt");
JsonGenerator jsonGenerator = factory.createGenerator(file, JsonEncoding.UTF8);
//對象開始
jsonGenerator.writeStartObject();
//寫入一個鍵值對
jsonGenerator.writeStringField("name", "小光");
//對象結(jié)束
jsonGenerator.writeEndObject();
//關(guān)閉jsonGenerator
jsonGenerator.close();
//讀取剛剛寫入的json
FileInputStream inputStream = new FileInputStream(file);
int i = 0;
final int SIZE = 1024;
byte[] buf = new byte[SIZE];
StringBuilder sb = new StringBuilder();
while ((i = inputStream.read(buf)) != -1) {
System.out.println(new String(buf,0,i));
}
inputStream.close();
//JsonParser解析的時候,思路是把json字符串根據(jù)邊界符分割為若干個JsonToken,這個JsonToken是一個枚舉類型。
//下面這個小例子,可以看出JsonToken是如何劃分類型的。
JsonParser parser = factory.createParser(s);
while (!parser.isClosed()){
JsonToken token = parser.currentToken();
System.out.println(token);
parser.nextToken();
}
JsonParser jsonParser = factory.createParser(s);
//下面是一個解析的實例
while (!jsonParser.isClosed()) {
JsonToken token = jsonParser.nextToken();
if (JsonToken.FIELD_NAME.equals(token)) {
String currentName = jsonParser.currentName();
token = jsonParser.nextToken();
if ("id".equals(currentName)) {
System.out.println("id:" + jsonParser.getValueAsInt());
} else if ("name".equals(currentName)) {
System.out.println("name:" + jsonParser.getValueAsString());
} else if ("array".equals(currentName)) {
token = jsonParser.nextToken();
while (!JsonToken.END_ARRAY.equals(token)) {
System.out.println("array:" + jsonParser.getValueAsString());
token = jsonParser.nextToken();
}
}
}
}
}
}
JackSon的常用注解
JackSon提供了一些的注解,可以用在類上或者是在字段上。通常是數(shù)據(jù)綁定的時候使用。下面幾個是最常用的幾個
@JsonInclude(Include.NON_EMPTY)
僅在屬性不為空時序列化此字段,對于字符串,即null或空字符串
@JsonIgnore
序列化時忽略此字段
@JsonProperty(value = “user_name”)
指定序列化時的字段名,默認使用屬性名
總結(jié)
JackSon使用起來還是十分方便的,提供的功能也很多,在使用的時候,需要結(jié)合自己的業(yè)務(wù)場景,選擇合適的解析方式。
參考資料
http://blog.lifw.org/post/63088058v
https://www.yiibai.com/jackson/
到此這篇關(guān)于淺談JackSon的幾種用法的文章就介紹到這了,更多相關(guān)JackSon 用法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
完美解決Logback configuration error detected的問題
這篇文章主要介紹了完美解決Logback configuration error detected的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08
Java8 stream 中利用 groupingBy 進行多字段分組求和案例
這篇文章主要介紹了Java8 stream 中利用 groupingBy 進行多字段分組求和案例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-08-08
Java項目之java+springboot+ssm實現(xiàn)理財管理系統(tǒng)設(shè)計
這篇文章主要介紹了Java項目java+springboot+ssm實現(xiàn)理財管理系統(tǒng)設(shè)計,使用了當(dāng)前較為流行的spring boot,spring,spring mvc,mybatis,shiro框架分頁處理使用了pagehelper進行操作,需要的朋友可以參考一下2022-03-03
Spring?循環(huán)依賴之AOP實現(xiàn)詳情
這篇文章主要介紹了Spring?循環(huán)依賴之AOP實現(xiàn)詳情,文章圍繞主題展開詳細的內(nèi)容介紹,具有一定的參考價值,需要的盆友可以參考一下2022-07-07
java使用stream判斷兩個list元素的屬性并輸出方式
這篇文章主要介紹了java使用stream判斷兩個list元素的屬性并輸出方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-06-06
SpringBoot前后端json數(shù)據(jù)交互的全過程記錄
現(xiàn)在大多數(shù)互聯(lián)網(wǎng)項目都是采用前后端分離的方式開發(fā),下面這篇文章主要給大家介紹了關(guān)于SpringBoot前后端json數(shù)據(jù)交互的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下2022-03-03
Java實現(xiàn)微信公眾號自定義菜單的創(chuàng)建方法示例
這篇文章主要介紹了Java實現(xiàn)微信公眾號自定義菜單的創(chuàng)建方法,結(jié)合實例形式分析了java創(chuàng)建微信公眾號自定義菜單的具體步驟、實現(xiàn)方法及相關(guān)操作注意事項,需要的朋友可以參考下2019-10-10

