Gson源碼分析——(壹)泛型

Gson是Google提供的一套用于解析Json數(shù)據(jù)的工具庫。本人之前其實(shí)寫過關(guān)于Gson源碼分析的文章,但由于當(dāng)時(shí)時(shí)間倉促,也沒能很好的整理筆頭思路,導(dǎo)致文章看起來有些雜亂?,F(xiàn)在靜下心來,重新審視一遍以前的作品。感覺還是有點(diǎn)遺漏,因此本人重新開始寫起這系列的文章。本篇是Gson解析的第一篇,依舊還是想從泛型開始講起,為什么呢?因?yàn)榉盒偷奶匦载灤┝苏麄€(gè)Gson的代碼。因?yàn)閱渭兊念惐旧砗茈y承擔(dān)多個(gè)類元數(shù)據(jù)的表現(xiàn)手段,而泛型的語法特性能很好的彌補(bǔ)這一點(diǎn)。(還插一些題外話,就是非墨后面研究的代碼是現(xiàn)在比較火的fastjson,并會(huì)通過兩個(gè)系列的文章比較兩個(gè)工具庫的差別和優(yōu)勢)。

1.泛型
泛型,是jdk提供的一種編譯期間檢查的機(jī)制。實(shí)際上,可以簡單理解為Java編譯器給你提供的一種編譯期類型檢查的機(jī)制?;蛟S你會(huì)說,我在運(yùn)行期如果不按照規(guī)范的類型,一樣會(huì)發(fā)生錯(cuò)誤。這是為什么呢?
這是因?yàn)?編譯器在泛型獲取的時(shí)候,注入了一條類型轉(zhuǎn)換語句。我們來看個(gè)例子:

//code1
List<String> list = new ArrayList<>(); 
try {           
Method m = list.getClass().getDeclaredMethod("add", new Class[]{Object.class});         m.invoke(list, 1);          
m.invoke(list, 2);      
} catch (Exception e) {         
System.out.println("error");        
} 

list.get();//error here

在code1中,我們?yōu)榱死@過編譯器檢查,我們使用反射注入的方式來往List里面注入非String對(duì)象:1和2。實(shí)際上,這個(gè)代碼在執(zhí)行的時(shí)候是不會(huì)報(bào)錯(cuò)的。這其實(shí)證明了泛型檢查是發(fā)生在編譯期。但是當(dāng)我用外部調(diào)用list.get()的時(shí)候,發(fā)生了錯(cuò)誤,為什么呢?我們來看下JVM到底做了什么?

//code2
95: iconst_0 
96: invokeinterface #56, 2; //InterfaceMethod java/util/List.get:(I)Ljava/lang/Object; 
101:    checkcast #62; //class java/lang/String

我們看到,當(dāng)我們調(diào)用接口方法List.get方法以后,虛擬機(jī)在101行執(zhí)行了一個(gè)checkcast指令,而這個(gè)指令我們并沒有在代碼中操作。因?yàn)檫@個(gè)指令并不是我們插入的,而是泛型聲明后,編譯器在執(zhí)行方法的時(shí)候,強(qiáng)行插入的。
相信通過上面的例子,大家對(duì)泛型已經(jīng)有了一個(gè)基本的概念,但是,看官么是否還有另外一個(gè)問題,就是既然泛型是存在于編譯期的東西,泛型數(shù)據(jù)是否會(huì)被存進(jìn)類文件中去呢?答案是會(huì),不然Gson就無法通過泛型來獲取最終生成的類型。

泛型繼承樹

在Java中,通過Type類來抽象Java中的類型。而類型可以大致分成兩大類:
1.Class類文件
2.泛型類型
泛型類型的子類一共有四種,分別是:
1.ParameterizedType: 參數(shù)化泛型參數(shù)
2.GenericArrayType: 泛型數(shù)組
3.TypeVariable: 泛型參數(shù)
4.WildcardType: 繼承型泛型

為了方便各位看官的理解,非墨用一個(gè)簡單的代碼例子來說明一下:

public class TypeClazz<T1, T2 extends Number> {

 public T2 member;
 public T1 member2;
 public Collection<? extends Number> collection;
 public Collection<T2> collection2;
public Collection<String> collection3;
 public T2[] array;
}

我們通過打印出來可以得到:

// code output
name:member->TypeVariableImpl
name:member2->TypeVariableImpl
name:collection->ParameterizedTypeImpl<WildcardTypeImpl>
name:collection2->ParameterizedTypeImpl<TypeVariableImpl>
name:collection3->ParameterizedTypeImpl<Class>
name:array->GenericArrayTypeImpl
name:method->Class(TypeVariableImpl,TypeVariableImpl)

通過上面反饋的結(jié)果,可以簡單理解為:
1.有<>包含的泛型屬于參數(shù)化類型ParameterizedType。比如collection*成員。
2.沒有尖括號(hào),直接用泛型標(biāo)記的變量分成兩種,一種是泛型變量,一種是通配符泛型
3.泛型數(shù)組是專門的類型GenericArrayType(如array變量)

相信上面的例子,已經(jīng)能很直觀的告訴大家,各種泛型代表的意義。但實(shí)際上,我們看到上面的繼承樹,我們并沒有完全解釋完,我們看到Class在實(shí)現(xiàn)Type接口的同時(shí),也實(shí)現(xiàn)了一個(gè)叫做GenericDeclaration的接口。我們來看下這個(gè)接口的定義:

public interface GenericDeclaration {

    /**
     * Returns the declared type parameters in declaration order. If there are
     * no type parameters, this method returns a zero length array.
     *
     * @return the declared type parameters in declaration order
     * @throws GenericSignatureFormatError
     *             if the signature is malformed
     */
    TypeVariable<?>[] getTypeParameters();
}

這個(gè)接口的作用有兩個(gè):
1.使用接口方法getTypeParameters來獲取泛型參數(shù)
2.使用這個(gè)接口來標(biāo)注,哪個(gè)類是支持泛型聲明的

我們通過上面的繼承樹可以看出,實(shí)現(xiàn)泛型聲明可以通過兩大類型來完成:
1.Class 類: 在類中進(jìn)行聲明。例如: class Clazz<T>
2.Executalbe:可執(zhí)行類,這個(gè)可執(zhí)行類分成兩個(gè)執(zhí)行子類,分別是Contructor和Method。也就是說,雖然Construtor的本質(zhì)就是一個(gè)方法,但是在Java中,還是將它歸結(jié)到一個(gè)單獨(dú)的類。( 例如: <T> void method(T param))

好的,帶著上面關(guān)于泛型的例子,我們可以進(jìn)入到Gson源碼的大門,相信各位看官跟著非墨品讀完之后,也會(huì)迷戀上這種小而精悍的代碼設(shè)計(jì)。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • object 變量可指向任何類的實(shí)例,這讓你能夠創(chuàng)建可對(duì)任何數(shù)據(jù)類型進(jìn)程處理的類。然而,這種方法存在幾個(gè)嚴(yán)重的問題...
    CarlDonitz閱讀 1,018評(píng)論 0 5
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 178,881評(píng)論 25 709
  • 夏天,圍觀群眾叫吃瓜群眾;春天,圍觀群眾就可以叫吹風(fēng)群眾;秋天,圍觀群眾就可以叫啃玉米群眾;冬天,圍觀群眾就可...
    左小樓閱讀 665評(píng)論 0 1
  • 1.展示的時(shí)候把代碼跑100遍2.解老詳情頁 native.send問題3.jenkins4.項(xiàng)目遷移
    以然以然閱讀 298評(píng)論 0 0

友情鏈接更多精彩內(nèi)容