欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解Java 中泛型的實現(xiàn)原理

 更新時間:2021年03月03日 14:41:12   作者:robothy  
這篇文章主要介紹了詳解Java 中泛型的實現(xiàn)原理,幫助大家更好的理解和學習使用Java,感興趣的朋友可以了解下

泛型是 Java 開發(fā)中常用的技術,了解泛型的幾種形式和實現(xiàn)泛型的基本原理,有助于寫出更優(yōu)質(zhì)的代碼。本文總結(jié)了 Java 泛型的三種形式以及泛型實現(xiàn)原理。

泛型

泛型的本質(zhì)是對類型進行參數(shù)化,在代碼邏輯不關注具體的數(shù)據(jù)類型時使用。例如:實現(xiàn)一個通用的排序算法,此時關注的是算法本身,而非排序的對象的類型。

泛型方法

如下定義了一個泛型方法, 聲明了一個類型變量,它可以應用于參數(shù),返回值,和方法內(nèi)的代碼邏輯。

class GenericMethod{
 public <T> T[] sort(T[] elements){
  return elements;
 }
}

泛型類

與泛型方法類似,泛型類也需要聲明類型變量,只不過位置放在了類名后面,作用的范圍包括了當前中的成員變量類型,方法參數(shù)類型,方法返回類型,以及方法內(nèi)的代碼中。

子類繼承泛型類時或者實例化泛型類的對象時,需要指定具體的參數(shù)類型或者聲明一個參數(shù)變量。如下,SubGenericClass 繼承了泛型類 GenericClass,其中類型變量 ID 的值為 Integer,同時子類聲明了另一個類型變量 E,并將E 填入了父類聲明的 T 中。

class GenericClass<ID, T>{
 
}

class SubGenericClass<T> extends GenericClass<Integer, T>{
 
}

泛型接口

泛型接口與泛型類類似,也需要在接口名后面聲明類型變量,作用于接口中的抽象方法返回類型和參數(shù)類型。子類在實現(xiàn)泛型接口時需要填入具體的數(shù)據(jù)類型或者填入子類聲明的類型變量。

interface GenericInterface<T> {
 T append(T seg);
}

泛型的基本原理

泛型本質(zhì)是將數(shù)據(jù)類型參數(shù)化,它通過擦除的方式來實現(xiàn)。聲明了泛型的 .java 源代碼,在編譯生成 .class 文件之后,泛型相關的信息就消失了??梢哉J為,源代碼中泛型相關的信息,就是提供給編譯器用的。泛型信息對 Java 編譯器可以見,對 Java 虛擬機不可見。

Java 編譯器通過如下方式實現(xiàn)擦除:

  • 用 Object 或者界定類型替代泛型,產(chǎn)生的字節(jié)碼中只包含了原始的類,接口和方法;
  • 在恰當?shù)奈恢貌迦霃娭妻D(zhuǎn)換代碼來確保類型安全;
  • 在繼承了泛型類或接口的類中插入橋接方法來保留多態(tài)性。

Java 官方文檔原文

Replace all type parameters in generic types with their bounds or Object if the type parameters are unbounded. The produced bytecode, therefore, contains only ordinary classes, interfaces, and methods.
Insert type casts if necessary to preserve type safety.
Generate bridge methods to preserve polymorphism in extended generic types.

下面通過具體代碼來說明 Java 中的類型擦除。

實驗原理:先用 javac 將 .java 文件編譯成 .class 文件,再使用反編譯工具 jad 將 .class 文件反編成回 Java 代碼,反編譯出來的 Java 代碼內(nèi)容反映的即為 .class 文件中的信息。

如下源代碼,定義 User 類,實現(xiàn)了 Comparable 接口,類型參數(shù)填入 User,實現(xiàn) compareTo 方法。

class User implements Comparable<User> {
 String name;
	
 public int compareTo(User other){
  return this.name.compareTo(other.name);
 }
}

JDK 中 Comparable 接口源碼內(nèi)容如下:

package java.lang;
public interface Comparable<T>{
 int compareTo(T o);
}

我們首先反編譯它的接口,Comparable 接口的字節(jié)碼文件,可以在 $JRE_HOME/lib/rt.jar 中找到,將它復制到某個目錄。使用 jad.exe(需要另外安裝)反編譯這個 Comparable.class 文件。

$ jad Comparable.class

反編譯出來的內(nèi)容放在 Comparable.jad 文件中,文件內(nèi)容如下:

// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.
// Jad home page: http://www.kpdus.com/jad.html
// Decompiler options: packimports(3)
// Source File Name: Comparable.java

package java.lang;

// Referenced classes of package java.lang:
//   Object

public interface Comparable
{

 public abstract int compareTo(Object obj);
}

對比源代碼 Comparable.java 和反編譯代碼 Comparable.jad 的內(nèi)容不難發(fā)現(xiàn),反編譯之后的內(nèi)容中已經(jīng)沒有了類型變量 T 。compareTo 方法中的參數(shù)類型 T 也被替換成了 Object。這就符合上面提到的第 1 條擦除原則。這里演示的是用 Object 替換類型參數(shù),使用界定類型替換類型參數(shù)的例子可以反編譯一下 Collections.class 試試,里面使用了大量的泛型。

使用 javac.exe 將 User.java 編譯成 .class 文件,然后使用 jad 將 .class 文件反編譯成 Java 代碼。

$ javac User.java
$ jad User.class

User.jad 文件內(nèi)容如下:

// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.
// Jad home page: http://www.kpdus.com/jad.html
// Decompiler options: packimports(3)
// Source File Name: User.java


class User
 implements Comparable
{

 User()
 {
 }

 public int compareTo(User user)
 {
  return name.compareTo(user.name);
 }

 // 橋接方法
 public volatile int compareTo(Object obj)
 {
  return compareTo((User)obj);
 }

 String name;
}

對比編輯的源代碼 User.java 和反編譯出來的代碼 User.jad,容易發(fā)現(xiàn):類型參數(shù)沒有了,多了一個無參構(gòu)造方法,多了一個 compareTo(Object obj) 方法,這個就是橋接方法,還可以發(fā)現(xiàn)參數(shù) obj 被強轉(zhuǎn)成 User 再傳入 compareTo(User user) 方法。通過這些內(nèi)容可以看到擦除規(guī)則 2 和規(guī)則 3 的實現(xiàn)方式。

強轉(zhuǎn)規(guī)則比較好理解,因為泛型被替換成了 Object,要調(diào)用具體類型的方法或者成員變量,當然需要先強轉(zhuǎn)成具體類型才能使用。那么插入的橋接方法該如何理解呢?

如果我們只按照下面方式去使用 User 類,這樣確實不需要參數(shù)類型為 Object 的橋接方法。

User user = new User();
User other = new User();
user.comparetTo(other);

但是,Java 中的多態(tài)特性允許我們使用一個父類或者接口的引用指向一個子類對象。

Comparable<User> user = new User();

而按照 Object 替換泛型參數(shù)原則,Comparable 接口中只有 compareTo(Object) 方法,假設沒有橋接方法,顯然如下代碼是不能運行的。所以 Java 編譯器需要為子類(泛型類的子類或泛型接口的實現(xiàn)類)中使用了泛型的方法額外生成一個橋接方法,通過這個方法來保證 Java 中的多態(tài)特性。

Comparable<User> user = new User();
Object other = new User();
user.compareTo(other);

而普通類中的泛型方法在進行類型擦除時不會產(chǎn)生橋接方法。例如:

class Dog{
 <T> void eat(T[] food){
 }
}

類型擦除之后變成了:

class Dog
{

 Dog()
 {
 }

 void eat(Object aobj[])
 {
 }
}

小結(jié)

Java 中的泛型有 3 種形式,泛型方法,泛型類,泛型接口。Java 通過在編譯時類型擦除的方式來實現(xiàn)泛型。擦除時使用 Object 或者界定類型替代泛型,同時在要調(diào)用具體類型方法或者成員變量的時候插入強轉(zhuǎn)代碼,為了保證多態(tài)特性,Java 編譯器還會為泛型類的子類生成橋接方法。類型信息在編譯階段被擦除之后,程序在運行期間無法獲取類型參數(shù)所對應的具體類型。

參考

https://docs.oracle.com/javase/tutorial/java/generics/index.html

https://stackoverflow.com/questions/25040837/generics-bridge-method-on-polymorphism

以上就是詳解Java 中泛型的實現(xiàn)原理的詳細內(nèi)容,更多關于Java 泛型實現(xiàn)原理的資料請關注腳本之家其它相關文章!

相關文章

  • Seata?AT模式TM處理流程圖文示例詳解

    Seata?AT模式TM處理流程圖文示例詳解

    這篇文章主要為大家介紹了Seata?AT模式TM處理流程圖文示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-09-09
  • Jmeter安裝的方法步驟詳解

    Jmeter安裝的方法步驟詳解

    這篇文章主要介紹了Jmeter安裝的方法步驟詳解,Apache JMeter是Apache組織開發(fā)的基于Java的壓力測試工具。用于對軟件做壓力測試,它最初被設計用于Web應用測試,但后來擴展到其他測試領域,需要的朋友可以參考下
    2019-07-07
  • SpringBoot應用部署于外置Tomcat容器的方法

    SpringBoot應用部署于外置Tomcat容器的方法

    這篇文章主要介紹了SpringBoot應用部署于外置Tomcat容器的方法,本文分步驟給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下
    2018-06-06
  • 實例講解Java設計模式編程中如何運用代理模式

    實例講解Java設計模式編程中如何運用代理模式

    這篇文章主要介紹了Java設計模式編程中如何運用代理模式,文中舉了普通代理和強制代理的例子作為代理模式的擴展內(nèi)容,需要的朋友可以參考下
    2016-02-02
  • 基于JVM性能監(jiān)控命令介紹

    基于JVM性能監(jiān)控命令介紹

    下面小編就為大家?guī)硪黄贘VM性能監(jiān)控命令介紹。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-09-09
  • Tomcat 實現(xiàn)WebSocket詳細介紹

    Tomcat 實現(xiàn)WebSocket詳細介紹

    這篇文章主要介紹了Tomcat 如何實現(xiàn)WebSocket的相關資料,對WebSocket協(xié)議通信的過程進行了詳細介紹,需要的朋友可以參考下
    2016-12-12
  • Java實現(xiàn)多叉樹和二叉樹之間的互轉(zhuǎn)

    Java實現(xiàn)多叉樹和二叉樹之間的互轉(zhuǎn)

    本文主要介紹了Java實現(xiàn)多叉樹和二叉樹之間的互轉(zhuǎn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-05-05
  • IntelliJ IDEA 2020.1.2激活工具下載及破解方法免費可用至2089年(強烈推薦)

    IntelliJ IDEA 2020.1.2激活工具下載及破解方法免費可用至2089年(強烈推薦)

    這篇文章主要介紹了IntelliJ IDEA 2020.1.2激活工具下載及破解方法免費可用至2089年(強烈推薦),本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-09-09
  • Jmeter測試時遇到的各種亂碼問題及解決

    Jmeter測試時遇到的各種亂碼問題及解決

    這篇文章主要介紹了Jmeter測試時遇到的各種亂碼問題及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • java中的Io(input與output)操作總結(jié)(三)

    java中的Io(input與output)操作總結(jié)(三)

    這一節(jié)我們來講Scanner類和PrintWriter類的用法,感興趣的朋友可以了解下
    2013-01-01

最新評論