Java 逆變與協變

来源:http://www.cnblogs.com/dw039/archive/2017/09/03/7459579.html
-Advertisement-
Play Games

最近一直忙於學習模電、數電,搞得頭暈腦脹,難得今天晚上擠出一些時間來分析一下Java中的逆變、協變。Java早於C#引入逆變、協變,兩者在與C#稍有不同,Java中的逆變、協變引入早於C#,故在形式沒有C#直觀(Google推出的基於jvm的Kotlin語音,則完全走向了C#的路線)。Java中逆變 ...


最近一直忙於學習模電、數電,搞得頭暈腦脹,難得今天晚上擠出一些時間來分析一下Java中的逆變、協變。Java早於C#引入逆變、協變,兩者在與C#稍有不同,Java中的逆變、協變引入早於C#,故在形式沒有C#直觀(Google推出的基於jvm的Kotlin語音,則完全走向了C#的路線)。Java中逆變、協變,在泛型集合使用中更多些、更直觀(像C#中的用法在Java中較少出現,但並非不可)。

正常泛型集合的使用

示例代碼如下:

public static void main(String[] args) {
        
        ArrayList<Number> alist = new ArrayList<>();
        alist.add(new Integer(10));
        alist.add(new Float(10.12));
        
        Number n = alist.get(0);
    }

對於alist集合而言,泛型類型參數 Number 已限定其可容納的類型。Integer/Float 類型的對象都可成功加入,並能通過get(index)獲取(獲取時註意類型為Number)。通過這個示例,可以看到正常泛型集合可 "存取規範的類型及其子類型對象"

協變

如下示例代碼:

public class Person {

}
public class Student extends Person{

}
public class Teacher extends Person{

}
public static void main(String[] args) { 
    ArrayList<Person> alist = new ArrayList<>(); 
    ArrayList<Student> slist = new ArrayList<>(); 
    //類型轉換錯誤 
    alist = slist; 
}

 

在 Java 中 Person雖是Student的父類,但泛型 ArrayList<Person> 卻並非 ArrayList<Student>的父類,所以 alist = slist 類型不相容。上面程式中的目的是用 slist 替代 alist的實際功能,因此在以後使用 alist 時實際調用的應該是 slist 對應的功能。

在 Java 中要滿足上述需求,應該具備2個條件:

1、放入數據時調用 alist.add(e) 要保證參數e,能轉換為 slist.add(e) 中所需類型

2、獲取數據時調用 alist.get(index) 的實際執行者 slist.get(index) 返回的結果能夠轉換為 alist 聲明的Person類型

上麵條件2是成立的,但條件1卻並不一定成立。因 ArrayList<Person> 規範類型為Person,Student、Teacher都滿足alist.add(e)的要求,卻並不滿足 slist.add(e)對類型必須是Student的要求。

修改代碼如下:

public static void main(String[] args) {
        
        ArrayList<? extends Person> alist = new ArrayList<>(); 
        ArrayList<Student> slist = new ArrayList<>(); 
        //類型轉換正常 
        alist = slist; 
        //添加數據錯誤
        alist.add(new Student());
        //正常
        Person p = alist.get(0);
    }

? extends Person 指明 alist 賦值的泛型集合約束類型只要是 extends Person 的就可以,因此將
ArrayList<Student> slist 賦值給 alist 滿足泛型類型參數約束。
alist.add(e) 根據 ? extend Person 的定義,e只要是 Person 類型或其子類型就可滿足添加條件,然而 slist 只允許 Student 類型對象加入,因此前面所說的條件1無法保證(null可以被加入,但卻意義) 。

因此Java中(根據前面2個條件分析可得到該結論)使用 extends 聲明的泛型或泛型集合,只能從其中取值,不能向其中添值
extends 聲明的集合只能用於從其中取出元素,可能有朋友會問“不能向其中放入值,又何來從其中取值”。看下麵的代碼:
public static void main(String[] args) {
        
            ArrayList<Student> slist = new ArrayList<>(); 
            slist.add(new Student());
            
            ArrayList<Teacher> tlist = new ArrayList<>(); 
            tlist.add(new Teacher());
            
            test(slist);
            test(tlist);      
    }
    private static void test(List<? extends Person> list) {
        for(Person p : list){
            System.out.println(p.toString());
        }
    }

通過上面的代碼很容易發現,test方法的參數 list 在方法內部不允許添加元素(null除外);但在 main 中卻可以向 slist、tlist 中添加元素,並作為實參傳遞到 test 方法的調用過程中。從 test 方法的角度分析,參數 list 裡面具體存放的是Person?Student?Teacher?無所謂,這裡統一當作Person對象來用絕對是類型安全的。這種把 泛型約束的子類型 當作 泛型約束的父類型 來用的情況就是協變。 

逆變 

示例代碼:

public static void main(String[] args) {
        ArrayList<Person> alist = new ArrayList<>(); 
        ArrayList<Student> slist = new ArrayList<>(); 
        //類型轉換錯誤 
        slist = alist; 
    }

如上代碼 意欲使用 alist 替代 slist 進行元素的存儲,跟進前面的分析上面代碼編譯錯誤。

在 Java 中要滿足上述需求,應該具備2個條件:

1、放入數據時調用 slist.add(e) 要保證參數e,能轉換為 alist.add(e) 中所需類型

2、獲取數據時調用 slist.get(index) 的實際執行者 alist.get(index) 返回的結果能夠轉換為 slist 聲明的Student類型

上麵條件1是成立的,但條件2卻並不一定成立。因 ArrayList<Person> 規範類型為Person,Student、Teacher都滿足alist.add(e)的要求,alist.get(index) 返回值卻並一定滿足 slist.get(index)返回值必須是Student類型的要求。

修改代碼如下:

public static void main(String[] args) {
        ArrayList<Object> alist = new ArrayList<>();
        ArrayList<? super Person> slist = new ArrayList<>();
        // 正常
        slist = alist;
        slist.add(new Student());
        slist.add(new Teacher());
        // 錯誤
        slist.add(new Object());

        Object obj = slist.get(0);
    }
? super Person 指明 slist 賦值的泛型集合約束類型只要是 super Person 的就可以,因此將 ArrayList<Object> alist 賦值給 slist 滿足泛型類型參數約束。

? super Person 雖指明集合可容納Person的父類類型(僅是有容納能力),但 Person 的繼承關係、層級並不明確(Person來自其他jar包)。因此該泛型約束對於Person的父類型並無約束能力,所以 super 禁止添加Person父類型到集合中(有能力容納,但不允許放入),只能放入Person及其子類型。
slist.get(index)調用實際上執行的是alist.get(index),最終返回值類型是Person?Teacher?Student? 不得而知,所以使用 super 聲明的泛型集合get返回只能為Object類型。
因此Java中(根據前面2個條件分析可得到該結論)使用 super 聲明的泛型或泛型集合,只能向其中添值,不要從其中取值(取回的值均為 Object 類型,沒有意義)

同樣有朋友會問 super 聲明的泛型集合只放不取有何意義?看下麵代碼:
public static void main(String[] args) {
        ArrayList<Person> alist = new ArrayList<>(); 
        test2(alist); 
        for(Person p : alist){
            System.out.println(p.toString());
        }
    }
    
    
    public static void test2(List<? super Person> list){
        list.add(new Student());
        list.add(new Teacher());
    }

 通過上面的代碼很容易發現,在 main 中 alist 有明確的類型,並作為實參傳遞到 test2 方法的調用過程中。從 test2 方法的角度分析,可以向集合中添加Person及其子類對象。這種把 泛型約束的父類型 當作 泛型約束的子類型 來用的情況就是逆變。 

 

說明:

Java 中使用逆變、協變的機會並不算多。如果要使用請記住:如果限定集合僅可放入值時用 super、如果要限定集合僅可取出值時用 extends、如果集合既需要放入值又要取出值時用標準泛型集合。

文章寫的倉促,若有不妥請各位朋友指正。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 特別聲明本隨筆copy於egon(林海峰)。 一 IO模型介紹 為了更好地瞭解IO模型,我們需要事先回顧下:同步、非同步、阻塞、非阻塞 同步(synchronous) IO和非同步(asynchronous) IO,阻塞(blocking) IO和非阻塞(non-blocking)IO分別是什麼,到底有 ...
  • 建立資料庫訪問類的封裝 <?php class DBDA { public $host = "localhost"; //伺服器地址 public $uid = "root"; //資料庫的用戶名 public $pwd = ""; //資料庫的密碼 public $dbname = "";//數據 ...
  • Spring與SpringMVC整合! 問:實際上SpringMVC就運行在Spring環境之下,還有必要整合麽?SpringMVC和Spring都有IOC容器,是不是都需要保留呢? 答案是:通常情況下,類似於數據源,事務,整合其他框架都是放在spring的配置文件中(而不是放在SpringMVC的 ...
  • 冒泡排序是一種非常常見的排序演算法。如同水中的一排泡泡,先冒出最大的一個泡泡。再冒出剩餘泡泡中的最大泡泡,依次類推,它的排序規則如下: 1. 從第一個元素開始,比較相鄰的兩個元素,如果後面的小於前面的,交換兩個的位置,一直比較到最後一個 2. 迴圈1中的操作,但已經確定的最大的元素不再參與比較 3. ...
  • 多線程 前言 我看了不止一個人說多線程是雞肋,但是就依照我個人覺得多線程在一些小型的爬蟲中還是可以顯著的提高速度的,相比多進程來說應該還是挺簡單的 使用多線程 繼承threading.Thread 繼承threading.Thread模塊是一個很好的一個選擇,就像java中也是可以繼承類和實現介面一 ...
  • Java類載入器算是一個老生常談的問題,大多Java工程師也都對其中的知識點倒背如流,最近在看源碼的時候發現有一些細節的地方理解還是比較模糊,正好寫一篇文章梳理一下。 關於Java類載入器的知識,網上一搜一大片,我自己也看過很多文檔,博客。資料雖然很多,但還是希望通過本文儘量寫出一些自己的理解,自己 ...
  • fileno()文件描述符 handle_request()處理單個請求 server_forever(poll_interval=0.5)處理多個請求,poll_interval每0.5秒檢測是否關閉, 作業:開發一個支持多用戶線上的FTP程式 要求: 1.用戶加密認證; 2.允許同時多用戶登錄; ...
  • 1 c() c表示”連接“(concatenate)。 在R中向量是連續存儲的,因此不能插入或刪除元素。 2 seq() seq()的特殊用法,可以用在for迴圈里for(i in seq()) 3 cumsum() 函數cumsum()它能計算向量的累計和(cumulative sums) 4 c ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...