jackson學習之八:常用方法註解

程式設計師欣宸發表於2021-02-05

歡迎訪問我的GitHub

https://github.com/zq2599/blog_demos

內容:所有原創文章分類彙總及配套原始碼,涉及Java、Docker、Kubernetes、DevOPS等;

系列文章彙總

本篇概覽

  • 本文是《jackson學習》系列的第八篇,繼續學習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資料夾下,如下圖紅框所示:

在這裡插入圖片描述

  1. jacksondemo是父子結構的工程,本篇的程式碼在annotation子工程中,裡面的methodannotation這個package下,如下圖:

在這裡插入圖片描述

JsonValue

  1. 在序列化時起作用,可以用來註解get方法或者成員變數;
  2. 一個類中,JsonValue只允許出現一次;
  3. 如果註解的是get方法,那麼該方法的返回值就是整個例項的序列化結果;
  4. 如果註解的是成員變數,那麼該成員變數的值就是整個例項的序列化結果;
  5. 下面是用來測試的Pojo類,JsonValue註解放在getField0方法上,此方法的返回值已經寫死了"abc":
    static class Test {

        private String field0;

        private String field1;

        @JsonValue
        public String getField0() { return "abc"; }

        public void setField0(String field0) { this.field0 = field0; }
        public String getField1() { return field1; }
        public void setField1(String field1) { this.field1 = field1; }
    }
  1. Test類的序列化結果如下,即getField0方法的返回值:

在這裡插入圖片描述

JsonCreator

  1. 在反序列化時,當出現有參構造方法時(可能是多個有參構造方法),需要通過JsonCreator註解指定反序列化時用哪個構造方法,並且在入參處還要通過JsonProperty指定欄位關係:
    static class Test {

        private String field0;
        private String field1;


        public Test(String field0) {
            this.field0 = field0;
        }

        // 通過JsonCreator指定反序列化的時候使用這個構造方法
        // 通過JsonProperty指定欄位關係
        @JsonCreator
        public Test(@JsonProperty("field0") String field0,
                    @JsonProperty("field1") String field1) {
            this.field0 = field0;
            this.field1 = field1;
        }

        @Override
        public String toString() {
            return "Test{" +
                    "field0='" + field0 + '\'' +
                    ", field1='" + field1 + '\'' +
                    '}';
        }
    }
  1. 反序列化結果如下:

在這裡插入圖片描述

JsonSetter

  1. JsonSetter註解在set方法上,被用來在反序列化時指定set方法對應json的哪個屬性;
  2. JsonSetter原始碼中,推薦使用JsonProperty來取代JsonSetter:

在這裡插入圖片描述
3. 測試程式碼和結果如下,可見反序列化時,是按照JsonSetter的value去json中查詢屬性的:

在這裡插入圖片描述

JsonGetter

  1. JsonGetter只能作為方法註解;
  2. 在序列化時,被JsonGetter註解的get方法,對應的json欄位名是JsonGetter的value;
  3. JsonGetter原始碼中,推薦使用JsonProperty來取代JsonGetter:

在這裡插入圖片描述
4. 測試程式碼和結果如下,可見序列化時JsonGetter的value會被作為json欄位名:

在這裡插入圖片描述

JsonAnyGetter

  1. JsonAnyGetter的作用有些特別:在序列化時,用Map物件的鍵值對轉成json的欄位和值;
  2. 理解JsonAnyGetter最好的辦法,是對比使用前後序列化結果的變化,先來看以下這段程式碼,是沒有JsonAnyGetter註解的,Test有兩個成員變數,其中map欄位是HashMap型別的:
package com.bolingcavalry.jacksondemo.annotation.methodannotation;

import com.fasterxml.jackson.annotation.JsonAnyGetter;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;

import java.util.HashMap;
import java.util.Map;

public class JsonAnySetterSerialization {

    static class Test {
        private String field0;
        private Map<String, Object> map;

        public String getField0() { return field0; }
        public void setField0(String field0) { this.field0 = field0; }
        public void setMap(Map<String, Object> map) { this.map = map; }
        public Map<String, Object> getMap() { return map; }
    }

    public static void main(String[] args) throws Exception {
        ObjectMapper mapper = new ObjectMapper();
        // 美化輸出
        mapper.enable(SerializationFeature.INDENT_OUTPUT);

        // 新增一個HashMap,裡面放入兩個元素
        Map<String, Object> map = new HashMap<>();
        map.put("aaa", "value_aaa");
        map.put("bbb", "value_bbb");

        Test test = new Test();
        test.setField0("000");

        // map賦值給test.map
        test.setMap(map);

        System.out.println(mapper.writeValueAsString(test));
    }
}
  1. 上述程式碼的執行結果如下,其實很好理解,就是field0和map兩個欄位而已:
{
  "field0" : "000",
  "map" : {
    "aaa" : "value_aaa",
    "bbb" : "value_bbb"
  }
}
  1. 接下來,對上述程式碼做一處改動,如下圖紅框所示,給getMap方法增加JsonAnyGetter註解:

在這裡插入圖片描述
5. 修改後的執行結果如下,原來的map欄位沒有了,map內部的所有鍵值對都成了json的欄位:

{
  "field0" : "000",
  "aaa" : "value_aaa",
  "bbb" : "value_bbb"
}
  1. 至此,可以品味出JsonAnyGetter的作用了:序列化時,將Map中的鍵值對全部作為JSON的欄位輸出

JsonAnySetter

  1. 弄懂了前面的JsonAnyGetter,對於JsonAnySetter的作用想必您也能大致猜到:反序列化時,對json中不認識的欄位,統統呼叫JsonAnySetter註解修飾的方法去處理;
  2. 測試的程式碼如下,Test類的setValue方法被JsonAnySetter註解,在反序列化時,json中的aaabbb欄位,都會交給setValue方法處理,也就是放入map中:
package com.bolingcavalry.jacksondemo.annotation.methodannotation;

import com.fasterxml.jackson.annotation.JsonAnySetter;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.ObjectMapper;

import java.util.HashMap;
import java.util.Map;

public class JsonAnySetterDeserialization {

    static class Test {

        private String field0;
        
        private Map<String, Object> map = new HashMap<>();

        @JsonAnySetter
        public void setValue(String key, Object value) {
            map.put(key, value);
        }

        @Override
        public String toString() {
            return "Test{" +
                    "field0='" + field0 + '\'' +
                    ", map=" + map +
                    '}';
        }
    }

    public static void main(String[] args) throws Exception {
        String jsonStr = "{\n" +
                "  \"field0\" : \"000\",\n" +
                "  \"aaa\" : \"value_aaa\",\n" +
                "  \"bbb\" : \"value_bbb\"\n" +
                "}";

        System.out.println(new ObjectMapper().readValue(jsonStr, Test.class));
    }
}
  1. 執行結果如下,可見aaa、bbb都被放入了map中:
Test{field0='null', map={aaa=value_aaa, field0=000, bbb=value_bbb}}
  1. 另外JsonAnySetter還可以作用在成員變數上,上面的程式碼中,去掉setValue方法,在成員變數map上增加JsonAnySetter註解,修改後如下,執行結果也是一模一樣的:
    static class Test {

        private String field0;

        @JsonAnySetter
        private Map<String, Object> map = new HashMap<>();

        @Override
        public String toString() {
            return "Test{" +
                    "field0='" + field0 + '\'' +
                    ", map=" + map +
                    '}';
        }
    }
  1. 注意,JsonAnySetter作用在成員變數上時,該成員變數必須是java.util.Map的實現類
  • 至此,Jackson常用註解已全部實戰完畢,希望這些豐富的註解能助您制定出各種靈活的序列化和反序列化策略;

你不孤單,欣宸原創一路相伴

  1. Java系列
  2. Spring系列
  3. Docker系列
  4. kubernetes系列
  5. 資料庫+中介軟體系列
  6. DevOps系列

歡迎關注公眾號:程式設計師欣宸

微信搜尋「程式設計師欣宸」,我是欣宸,期待與您一同暢遊Java世界...
https://github.com/zq2599/blog_demos

相關文章