Kubernetes官方java客戶端之二:序列化和反序列化問題

歡迎訪問個人GitHub

https://github.com/zq2599/blog_demos前端

內容:全部原創文章分類彙總及配套源碼,涉及Java、Docker、Kubernetes、DevOPS等;java

問題場景

本文是《Kubernetes官方java客戶端》的第二篇,在進入編碼實戰章節以前,有個問題須要你們有足夠的瞭解,避免在後面的實戰中耗費精力處理此類問題,來看看到底是什麼問題:git

  1. SpringBoot是經常使用的應用框架,《Kubernetes官方java客戶端》系列的應用都是基於SpringBoot-2.3.1版本的;程序員

  2. 下圖是SpringBoot-2.3.1.RELEASE的官方文檔,紅框代表默認的JSON處理庫是Jackson:
    在這裏插入圖片描述github

  3. 看到這裏您是否有種不祥預感:K8S官方java客戶端是谷歌的,涉及到JSON處理時會不會首選自家的Gson?web

  4. V1HTTPGetAction.java是java客戶端中經常使用到的數據結構,用來封裝http請求相關的參數,來看看其源碼,以下圖,果真用上了Gson的註解:
    在這裏插入圖片描述spring

  5. 上圖提到的IntOrString類要重點關注,用處普遍,打開其源碼以下圖,請記下紅框2中的代碼,後面提到的問題就來源於此:
    在這裏插入圖片描述shell

  • 小結:SpringBoot默認的JSON處理類是Jackson,K8S官方java客戶端內的Bean在涉及到JSON相關的序列化和反序列化處理時,使用了Gson註解,所以上述Bean實例在SpringBoot中涉及到JSON處理時,可能會有問題(這時只能說可能),例如RestController返回對象,會被Jackson轉爲JSON;

復現問題

  1. 這裏用一個SpringBoot工程來演示此問題(該工程名爲OutsideclusterApplication,下一篇文章會詳細說明),以下代碼是個http接口響應,可見V1PodList實例做爲接口返回時,會被SpringBoot用Jackson轉爲JSON返回給前端:
@RequestMapping(value = "/hello")
    public V1PodList hello() throws Exception {
        // 存放K8S的config文件的全路徑
        String kubeConfigPath = "/Users/zhaoqin/temp/202007/05/config";

        // 以config做爲入參建立的client對象,能夠訪問到K8S的API Server
        ApiClient client = ClientBuilder
                .kubeconfig(KubeConfig.loadKubeConfig(new FileReader(kubeConfigPath)))
                .build();

        Configuration.setDefaultApiClient(client);

        CoreV1Api api = new CoreV1Api();

        // 調用客戶端API取得全部pod信息
        V1PodList v1PodList = api.listPodForAllNamespaces(null, null, null, null, null, null, null, null, null);

        return v1PodList;
    }
  1. 上述代碼運行起來,在瀏覽器訪問該接口時,控制檯拋出如下錯誤,IntOrString.getStrValue方法,就是前面我們看過的那段,IntOrString中實際上保存的是int數據,可是Jackson執行了其getStrValue方法:
    在這裏插入圖片描述數據庫

  2. 至於爲何Jackson會執行getStrValue方法,篇幅緣由就不在此展開了,簡單提一下,在java客戶端的BeanPropertyWriter類中,選擇方法的邏輯以下圖,紅框中展現了斷定邏輯,此處getStrValue方法命中了該邏輯,若是您嘗試用在紅框處打上斷點觀察,會發現有不少方法都符合此條件:
    在這裏插入圖片描述json

解決問題的思路

我這裏,解決問題的思路有兩個:

  1. 讓Jackson在序列化的時候,可以調用正確的方法,以IntOrString爲例,若是此時內部保存int型數據,就應該執行其getIntValue方法便可;
  2. Bean中使用了Gson註釋,就是打算用Gson來處理序列化和反序列化操做的,所以序列化和反序列化的地方都改用Gson處理;
  • 上述兩個思路,我選擇了第二種,畢竟第一種太難了...

解決問題

  1. 問題解決起來並不難,先看SpringBoot-2.3.1.RELEASE官方文檔:
    在這裏插入圖片描述

  2. 結合官方文檔,咱們要作兩件事情:

  • 首先,classpath中有Gson,這個已經有了,由於K8S官方java客戶端會依賴Gson;
  • 其次,classpath中不要出現Jackson,爲了達到這個目的咱們須要作如下操做,排除spring-boot-starter-web的依賴(爲何不直接排除jackson的庫呢?您能夠執行mvn dependency:tree命令細看依賴樹,會發現對jackson的依賴並不是單一關係):
<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-web</artifactId>
		<exclusions>
			<exclusion>
				<groupId>org.springframework.boot</groupId>
				<artifactId>spring-boot-starter-json</artifactId>
			</exclusion>
		</exclusions>
</dependency>
  1. 建議您執行mvn dependency:tree命令細看整個項目的依賴樹,確保jackson依賴已經所有去掉;
  2. 再次運行上述項目,以下圖,服務端再也不報錯,頁面上返回數據正常:
    在這裏插入圖片描述

使用Jackson的場景

  • 上述方式雖然可行,但並不是全部項目都能堅持使用Gson而放棄Jackson,對於使用Jackson的項目,請避免Jackson參與K8S官方java客戶端bean的序列化和反序列化操做,以上面出現的Controller代碼爲例,不要直接將V1PodList實例返回,您能夠選擇先用Gson序列化成JSON字符串,再返回字符串給前端,也能夠本身定義VO對象,將V1PodList實例轉成VO對象再返回;

  • 至此,使用K8S官方java客戶端以前要注意的問題已經弄明白了,接下來的進入精彩的實戰章節吧,一塊兒體驗kubernetes官方爲java程序員精心準備的工具;

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

  1. Java系列
  2. Spring系列
  3. Docker系列
  4. kubernetes系列
  5. 數據庫+中間件系列
  6. DevOps系列

歡迎關注公衆號:程序員欣宸

微信搜索「程序員欣宸」,我是欣宸,期待與您一同暢遊Java世界...
https://github.com/zq2599/blog_demos

相關文章
相關標籤/搜索