Java核心技術梳理-泛型

来源:https://www.cnblogs.com/yuanqinnan/archive/2019/04/29/10789988.html
-Advertisement-
Play Games

一、引言 在學習集合的時候我們會發現一個問題,將一個對象丟到集合中後,集合併不記住對象的類型,統統都當做Object處理,這樣我們取出來再使用時就得強制轉換類型,導致代碼臃腫,而且加入集合時都是以Object,沒做類型檢查,那麼強制轉換就容易出錯,泛型的誕生就是為解決這些問題。 二、使用泛型 泛型是 ...


一、引言

在學習集合的時候我們會發現一個問題,將一個對象丟到集合中後,集合併不記住對象的類型,統統都當做Object處理,這樣我們取出來再使用時就得強制轉換類型,導致代碼臃腫,而且加入集合時都是以Object,沒做類型檢查,那麼強制轉換就容易出錯,泛型的誕生就是為解決這些問題。

二、使用泛型

泛型是如何解決這個問題呢?按照上面的問題,我們只需要在創建集合時指定集合元素的類型,那麼集合就能記住對象的類型,那當我們加入是就只能按照指定的類型進行加入,而取出使用時也不需要強制轉換:

ArrayList<Integer> list = new ArrayList<Integer>();

這是Java 7之前的寫法,很明顯構造器上面的泛型沒有必要,現在推薦以下寫法:

ArrayList<Integer> list = new ArrayList<>();

既然我已經指定了類型,那麼添加時只能添加Integer,並且使用時可以直接當做Integer使用

System.out.println(list.get(2)+3);

這種參數化類型就是泛型,泛型就是允許在定義類、介面、方法時使用類型形參,這個參數形參將在申明變數、創建對象、調用方法時動態指定。

三、 定義泛型介面、類

來看看List介面和Map介面的定義:

public interface List<E> extends Collection<E>
public interface Map<K,V>

List介面定義時指定了一個類型形參,Map介面定義了兩個類型形參。介面定義了形參之後,在介面中形參就可以當做一種類型來使用,那麼其中的方法就可以使用類型形參

boolean add(E e);

這種方式其實也是一種代碼復用,我們通過類型形參,高度的將參數抽象,而不需要每種類型都去重新定義類,只要在使用時確定類型即可。我們也可以自定義泛型類

public class WebResult<T> {
    //使用T類型形參定義實例變數
    private T data;

    public WebResult() {
    }
    //使用T類型形參構造對象
    public WebResult(T data) {
        this.data = data;
    }

    public void setData(T data) {
        this.data = data;
    }

    public T getData() {
        return this.data;
    }

    public static void main(String[] args) {
        WebResult<String> webResult = new WebResult<>("返回一個String對象");
        System.out.println(webResult.getData());
        WebResult<Integer> webResult1 = new WebResult<>(10);
        System.out.println(webResult1.getData());
    }

}

四、通配符

先看下麵這段代碼:

public static void main(String[] str){
    ArrayList<String> arrayList=new ArrayList();
    test(arrayList);
}

public static void test(List<Object> test){
    for (int i = 0; i <test.size() ; i++) {
        System.out.println(test.get(i));
    }
}

這段代碼會出現編譯錯誤,因為List<String>對象不能作為List<Object>使用,這說明泛型不是協變的,因為之前數組的設計是協變的,導致存在安全性問題,而泛型的設計原則是編譯時不出現警告就不會出現類型轉換錯誤,那為了表示各種泛型的父類,就引入了通配符:?這個問號代表可以是匹配任何類型。

將方法修改:

public static void test(List<?> test){
    for (int i = 0; i <test.size() ; i++) {
        System.out.println(test.get(i));
    }
}

這樣便可以順利編譯,我們再加上這段代碼:

public static void main(String[] str){
    ArrayList<String> arrayList=new ArrayList();
    test(arrayList);
    List<?> strings = arrayList;
    strings.add("abc");
}

這裡我們可以將arrayList給strings,按說這個時候是不能賦值的,因為List不知道類型參數的值,這是編譯器作用,可以進行類型推理,但是後面的strings.add("abc")是不能通過編譯的,編譯器不能對 List 的類型參數作出足夠嚴密的推理,以確定將 String 傳遞給 List.add() 是類型安全的。所以編譯器將不允許這麼做。

4.1 設置通配符上限

List<?>這種方式,通配的是所有的類型,很多時候我們可以確定是哪一類對象可以添加進去,我們只希望它代表某一類泛型的父類,這個時候我們可以設置通配符的上限。

//動物類
public abstract class Animal {
    public abstract void say();
}

public class Cat extends Animal {
    @Override
    public void say() {
        System.out.println("喵喵");
    }
}

public class Dog extends Animal {

    @Override
    public void say() {
        System.out.println("旺旺");
    }
}

這個時候我們就限定了上限

public static void test1(List<? extends Animal> animals) {
    for (int i = 0; i < animals.size(); i++) {
        animals.get(i).say();
    }
}

我們也可以直接在定義類型形參的時候設置上限

public class WebResult<T extends Animal> {

4.2 通配符下限

既然有設置上限,那也有設置下限,那在什麼情況下會使用下限呢?看個例子

//將src中的集合複製到dest,並返回最後一個值
public static <T> T copy(Collection<T> dest, Collection<? extends T> src) {
    T last = null;
    for (T ele : src) {
        last = ele;
        dest.add(ele);
    }
    return last;
}

功能比較很簡單,將src中的集合複製到dest,並返回src最後一個值

List<Number> ln = new ArrayList<>();
List<Integer> li = new ArrayList<>();
//編譯出錯,類型不確定
Integer last = copy(ln, li);

這個時候出錯,因為雖然我們知道返回的值一定是Integer,但是由於copy方法的返回值並不是,所有相當於我們在複製的過程中丟失了src的類型,如果我們想定義約束關係使得返回值明確即:dest集合元素類型與src的關係要麼相同要麼是其父類,為了表示這種約束關係,引入了<? super T> 這個通配符表示它必須是T本身或者T的父類。

//將src中的集合複製到dest,並返回最後一個值
public static <T> T copy(Collection<? super T> dest, Collection<? extends T> src) {
    T last = null;
    for (T ele : src) {
        last = ele;
        dest.add(ele);
    }
    return last;
}

五、泛型方法

除了泛型介面和泛型類,我們還可以定義泛型方法,寫法如下:

static <T> void arrayToList(T[] a, List<T> list) {
    for (T o : a) {
        list.add(o);
    }
}

調用如下:

Object[] objects = new Object[10];
List<Object> list = new ArrayList<>();
arrayToList(objects, list);

Integer[] integers = new Integer[10];
List<Integer> integerList = new ArrayList();
arrayToList(integers, integerList);

String[] strings = new String[10];
List<String> stringList = new ArrayList<>();
arrayToList(strings, stringList);
//編譯錯誤,類型不正確
arrayToList(strings, integerList);

這裡可以看出泛型方法跟類型通配符的功能有點類似,其實在大部分情況下我們可以用泛型方法代替類型通配符。

泛型方法允許類型形參被用來表示方法的一個或者多個參數之間的依賴關係,或者說與返回值之間的關係,如果沒有這種關係,我們就不使用泛型方法。

六、擦除與轉換

當把一個具有泛型信息的對象賦給一個沒有泛型信息的變數時,所有的類型信息就都丟掉了,比如List<String>類型被轉換成List,則對該List的類型檢查也變成了Object。

public class WebResult<T extends Number> {
    //使用T類型形參定義實例變數
    private T data;

    public WebResult() {
    }

    //使用T類型形參構造對象
    public WebResult(T data) {
        this.data = data;
    }

    public void setData(T data) {
        this.data = data;
    }

    public T getData() {
        return this.data;
    }

    public static void main(String[] args) {
        WebResult<Integer> webResult1 = new WebResult<>(10);
        System.out.println(webResult1.getData());
        WebResult<Integer> a = new WebResult<>(20);
        WebResult b = a;
        //已經擦除了泛型,只能按最高類型Object
        //Integer bData = b.getData();
        Object object=b.getData();
    }

}

原本的泛型類上限是Number,而當把a賦給擦除泛型的b對象時,編譯器失去了推斷能力,只能把其當做Objec來處理。

而當一個List轉成泛型對象是java是允許的

List<Integer> integerList = new ArrayList<>();
List stringList = integerList;
//允許直接將list對象轉換給
List<String> strings = stringList;
//直接獲取數據會出現錯誤,因為轉換不成功
System.out.println(stringList.get(0));

 


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

-Advertisement-
Play Games
更多相關文章
  • 一、Python 簡介 Python定義:是一個免費、開源、跨平臺、動態、面向對象的編程語言。 Python程式的執行(運行)方式有兩種:互動式、文件式 互動式在命令行輸入指令,回城即可得到結果。1.打開終端2.進行互動式:python33.編寫代碼:print(“hello world”)4.離開 ...
  • [TOC] 有名函數(掌握) 我們之前定的函數都是有名函數,它是基於函數名使用。 from func from func from func 匿名函數(掌握) 匿名函數,他沒有綁定名字,使用一次即被收回,加括弧既可以運行。 (x, y) 3 與內置函數聯用(掌握) 匿名函數通常與max()、min( ...
  • Python基礎之容器類型的公共方法,包括 Python內置函數,切片,運算符,完整的for迴圈。其中,Python內置函數包括 內置函數羅列,內置函數使用;切片僅 包含切片的定義和使用;運算符包括 運算符羅列,運算符的使用;完整的for迴圈 包括 完整的for迴圈語法,for迴圈演示,break打... ...
  • 1.包裝類 把八大基本數據類型封裝到一個類中(包裝類),並提供屬性和方法。讓我們更加方便的操作基本數據類型。但包裝類的出現並不是為了取代基本數據類型,也取代不了。 包裝類位於java.lang包中。 Number 類 Number數值類型是byte、double、float、int、long 和 s ...
  • 什麼是雙端隊列? ArrayDeque是怎麼實現雙端隊列的? ArrayDeque是線程安全的嗎? ArrayDeque是有界的嗎? ...
  • 新聞 ".NET版本的Apache Spark" "Apache Spark預覽版介紹" "F Apache Spark示例" "微軟Build 2019大會(5月6日至8日)" "Rider用於F 的解決方案內的重命名" "Spark+AI峰會——開發智能雲與智能邊緣" "CNTK最新的重大發佈— ...
  • 1.常量是什麼?有什麼存在的意思? 答:舉個例子,公司開發,資料庫的地址用戶名密碼等信息一般固定不變,不需要後面程式改動。 如果用變數,$db = 'xx';其他人寫程式,後面好巧不巧,修改了這個變數,是不是就出問題了。 再比如,程式中要用到圓周率等,用變數存,就怕哪天被修改了,程式上是不會報錯的, ...
  • 1. swagger知識點補充 1.1. 概述 1. 在swagger的使用過程中,除了網上常見的例子,還會有很多細節上的東西需要註意和改寫,這裡我列幾點我使用過程中遇到的問題和改進方式 1.2. 知識點 1.2.1. 模型例子 1. 我們在進行POST的請求的時候,尤其是增加一條數據,我們往往會有 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...