詳解Java泛型中類型擦除問(wèn)題的解決方法
以前就了解過(guò)Java泛型的實(shí)現(xiàn)是不完整的,最近在做一些代碼重構(gòu)的時(shí)候遇到一些Java泛型類型擦除的問(wèn)題,簡(jiǎn)單的來(lái)說(shuō),Java泛型中所指定的類型在編譯時(shí)會(huì)將其去除,因此List 和 List 在編譯成字節(jié)碼的時(shí)候?qū)嶋H上是一樣的。因此java泛型只能做到編譯期檢查的功能,運(yùn)行期間就不能保證類型安全。我最近遇到的一個(gè)問(wèn)題如下:
假設(shè)有兩個(gè)bean類
/** Test. */ @Data @NoArgsConstructor @AllArgsConstructor public static class Foo { public String name; } /** Test. */ @Data @NoArgsConstructor @AllArgsConstructor public static class Dummy { public String name; }
以及另一個(gè)對(duì)象
@NoArgsConstructor @AllArgsConstructor @Data public static class Spec<T> { public String spec; public T deserializeTo() throws JsonProcessingException { var mapper = new ObjectMapper(); return (T) mapper.readValue(spec, Foo.class); } }
可以看到Spec
對(duì)象中保存了以上兩種類型json序列化后的字符串,并提供了方法將string spec 反序列化成相應(yīng)的類型,比較理想的方式是在反序列化的方法中能夠獲取到參數(shù)類型 T 的實(shí)際類型,理論上運(yùn)行時(shí)Spec類型是確定了,因此T也應(yīng)該是確定的,但是因?yàn)轭愋筒脸?,所以?shí)際上獲取不到他的類型。
按照以下嘗試 通過(guò)((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()獲取泛型類型,經(jīng)過(guò)測(cè)試是獲取不到的
@Test public void test() throws JsonProcessingException { var foo = new Foo("foo"); var spec = new Spec<Foo>(mapper.writeValueAsString(foo)); var deserialized = spec.deserializeTo(); Assertions.assertTrue(deserialized instanceof Foo); } @NoArgsConstructor @AllArgsConstructor @Data public static class Spec<T> { public String spec; private Class<T> getSpecClass() { return (Class<T>) ((ParameterizedType) getClass().getGenericSuperclass()) .getActualTypeArguments()[0]; } public T deserializeTo() throws JsonProcessingException { var mapper = new ObjectMapper(); System.out.println(spec); return (T) mapper.readValue(spec, getSpecClass()); } }
會(huì)有以下的錯(cuò)誤
java.lang.ClassCastException: class java.lang.Class cannot be cast to class java.lang.reflect.ParameterizedType (java.lang.Class and java.lang.reflect.ParameterizedType are in module java.base of loader 'bootstrap')
有兩種辦法來(lái)繞過(guò)這個(gè)問(wèn)題
第一種比較簡(jiǎn)單,就是在創(chuàng)建spec對(duì)象時(shí),直接把類型的class傳進(jìn)來(lái),這樣就可以直接使用。
第二種是創(chuàng)建spec的子類中使用這個(gè)方法就可以獲取泛型的類型
@Data public abstract static class AbstractSpec<T> { public String spec; public AbstractSpec(String spec) { this.spec = spec; } private Class<T> getSpecClass() { return (Class<T>) ((ParameterizedType) getClass().getGenericSuperclass()) .getActualTypeArguments()[0]; } public T deserializeTo() throws JsonProcessingException { var mapper = new ObjectMapper(); System.out.println(spec); return (T) mapper.readValue(spec, getSpecClass()); } } public static class Spec extends AbstractSpec<Foo> { public Spec(String spec) { super(spec); } } @Test public void test() throws JsonProcessingException { var foo = new Foo("foo"); var spec = new Spec(mapper.writeValueAsString(foo)); var deserialized = spec.deserializeTo(); Assertions.assertTrue(deserialized instanceof Foo); }
這里spec類就可以順利的被反序列化。
這個(gè)和最開(kāi)始失敗的case的差別就是新增了一個(gè)子類,主要的差別是getGenericSuperclass的返回值有差異,非子類的情況下,獲取到的是Object。
因此理論上子類Spec的類型信息中,實(shí)際上是保存了父類中的類型參數(shù)信息的,也就是例子中的Foo. 按照 https://stackoverflow.com/questions/42874197/getgenericsuperclass-in-java-how-does-it-work 的方式,可以查看到Spec類的字節(jié)碼中有相應(yīng)的類型信息。
$ javap -verbose ./org/apache/flink/kubernetes/operator/controller/GenericTest\$Spec.class | grep Signature #15 = Utf8 Signature Start Length Slot Name Signature Signature: #19 // Lorg/apache/flink/kubernetes/operator/controller/GenericTest$AbstractSpec<Lorg/apache/flink/kubernetes/operator/controller/GenericTest$Foo;>;
到此這篇關(guān)于詳解Java泛型中類型擦除問(wèn)題的解決方法的文章就介紹到這了,更多相關(guān)Java泛型類型擦除內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳解java實(shí)現(xiàn)簡(jiǎn)單掃碼登錄功能(模仿微信網(wǎng)頁(yè)版掃碼)
這篇文章主要介紹了java實(shí)現(xiàn)簡(jiǎn)單掃碼登錄功能(模仿微信網(wǎng)頁(yè)版掃碼),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-05-05簡(jiǎn)談java并發(fā)FutureTask的實(shí)現(xiàn)
這篇文章主要介紹了簡(jiǎn)談java并發(fā)FutureTask的實(shí)現(xiàn),FutureTask都是用于獲取線程執(zhí)行的返回結(jié)果。文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,,需要的朋友可以參考下2019-06-06Java環(huán)境中MyBatis與Spring或Spring MVC框架的集成方法
和MyBatis類似,Spring或者Spring MVC框架在Web應(yīng)用程序的運(yùn)作中同樣主要負(fù)責(zé)處理數(shù)據(jù)庫(kù)事務(wù),這里我們就來(lái)看一下Java環(huán)境中MyBatis與Spring或Spring MVC框架的集成方法2016-06-06java通過(guò)Jsoup爬取網(wǎng)頁(yè)過(guò)程詳解
這篇文章主要介紹了java通過(guò)Jsoup爬取網(wǎng)頁(yè)過(guò)程詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09MPAndroidChart開(kāi)源圖表庫(kù)的使用介紹之餅狀圖、折線圖和柱狀圖
這篇文章主要介紹了MPAndroidChart開(kāi)源圖表庫(kù)的使用介紹之餅狀圖、折線圖和柱狀圖的相關(guān)資料,需要的朋友可以參考下2016-02-02基于Java實(shí)現(xiàn)中文分詞系統(tǒng)的示例代碼
這篇文章主要為大家詳細(xì)介紹了如何利用Java語(yǔ)言實(shí)現(xiàn)一個(gè)簡(jiǎn)易的中文分詞系統(tǒng),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以嘗試一下2022-07-07詳解java數(shù)組進(jìn)行翻轉(zhuǎn)的方法有哪些
這篇文章主要介紹了詳解java數(shù)組進(jìn)行翻轉(zhuǎn)的方法有哪些,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01Spring Boot 2.x 實(shí)現(xiàn)文件上傳功能
這篇文章主要介紹了Spring Boot 2.x 實(shí)現(xiàn)文件上傳功能,本文分步驟通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-01-01springboot配置數(shù)據(jù)庫(kù)密碼特殊字符報(bào)錯(cuò)的解決
這篇文章主要介紹了springboot配置數(shù)據(jù)庫(kù)密碼特殊字符報(bào)錯(cuò)的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02