java基礎——泛型

来源:https://www.cnblogs.com/malinyan/archive/2023/07/30/17592372.html
-Advertisement-
Play Games

## 泛型的引入 看下麵這段代碼: ```java private static int add(int a, int b) { System.out.println(a + "+" + b + "=" + (a + b)); return a + b; } private static float ...


泛型的引入

看下麵這段代碼:

private static int add(int a, int b) {
    System.out.println(a + "+" + b + "=" + (a + b));
    return a + b;
}

private static float add(float a, float b) {
    System.out.println(a + "+" + b + "=" + (a + b));
    return a + b;
}

private static double add(double a, double b) {
    System.out.println(a + "+" + b + "=" + (a + b));
    return a + b;
}

如果沒有泛型,要實現不同類型的加法,每種類型都需要重載一個add方法;通過泛型,我們可以復用為一個方法:

private static <T extends Number> double add(T a, T b) {
    System.out.println(a + "+" + b + "=" + (a.doubleValue() + b.doubleValue()));
    return a.doubleValue() + b.doubleValue();
}

泛型中的類型在使用時指定,不需要強制類型轉換(類型安全,編譯器會檢查類型

下麵這段代碼:

List list = new ArrayList();
list.add("xxString");
list.add(100d);
list.add(new Person());

我們在使用上述list中,list中的元素都是Object類型(無法約束其中的類型),所以在取出集合元素時需要人為的強制類型轉化到具體的目標類型,且很容易出java.lang.ClassCastException`異常。

引入泛型,它將提供類型的約束,提供編譯前的檢查:

List<String> list = new ArrayList<String>();

// list中只能放String, 不能放其它類型的元素

泛型的好處

1、編譯時,檢查添加元素的類型,提高了安全性

2、減少了類型轉換的次數,提高效率

3、不再提示編譯警告

介紹

1.泛型又稱參數化類型,是Jdk5.0出現的新特性,解決數據類型的安全性問題

2.在類聲明或實例化時只要指定好需要的具體的類型即可。

3.Java泛型可以保證如果程式在編譯時沒有發出警告,運行時就不會產生ClassCastException.異常。同時,代碼更加簡潔、健壯

4.泛型的作用是:可以在類聲明時通過一個標識表示類中某個屬性的類型,或者是某個方法的返回值的類型,或者是參數類型。

代碼舉例:

public class Generic03 {
    public static void main(String[] args) {
        //註意,特別強調:E具體的數據類型在定義Person對象的時候指定,即在編譯期間,就確定E是什麼類型
        Person<String> person = new Person<>("dfdfs");
        person.show();//String

        /**
         * Person類相當於下麵這樣子
         * class Person{
         *     String s;//E表示s的數據類型,該數據類型在定義Person對象的時候指定,即在編譯期間,就確定E是什麼類型
         *
         *     public Person(String s) {//E也可以是參數類型
         *         this.s = s;
         *     }
         *
         *     public String f(){//返回類型使用E
         *         return s;
         *     }
         *     public void show(){
         *         System.out.println(s.getClass());//顯示s的運行類型
         *     }
         * }
         */

        Person<Integer> person2 = new Person<>(100);
        person2.show();

        /**
         * Person類相當於下麵這樣子
         * class Person{
         *     Integer s;//E表示s的數據類型,該數據類型在定義Person對象的時候指定,即在編譯期間,就確定E是什麼類型
         *
         *     public Person(Integer s) {//E也可以是參數類型
         *         this.s = s;
         *     }
         *
         *     public Integer f(){//返回類型使用E
         *         return s;
         *     }
         *     public void show(){
         *         System.out.println(s.getClass());//顯示s的運行類型
         *     }
         * }
         */
    }
}
//泛型的作用是:可以在類聲明時通過一個標識表示類中某個屬性的類型,或者是某個方法的返回值的類型,或者是參數類型
class Person<E>{
    E s;//E表示s的數據類型,該數據類型在定義Person對象的時候指定,即在編譯期間,就確定E是什麼類型

    public Person(E s) {//E也可以是參數類型
        this.s = s;
    }

    public E f(){//返回類型使用E
        return s;
    }
    public void show(){
        System.out.println(s.getClass());//顯示s的運行類型
    }
}

語法

泛型的聲明

interface 介面名{}和class 類名<K,V>{}

說明:

1)其中,T,K,V不代表值,而是表示類型

2)任意字母都可以。常用T表示,是Type的縮寫

泛型的實例化

要在類名後面指定類型參數的值(類型)。如:

List<String> strList = new ArrayList<String>();
Iterator<Customer> iterator = customers.iterator();

泛型使用舉例

需求:

​ 1.創建3個學生對象

​ 2.放入到HashSet中學生對象,使用.

​ 3.放入到HashMap中,要求Key 是String name,Value就是學生對象

​ 4.使用兩種方式遍歷

public class GenericExercise {
    public static void main(String[] args) {
        //使用泛型方式給HashSet放入3個學生對象
        HashSet<Student> students = new HashSet<>();
        students.add(new Student("jack",18));
        students.add(new Student("tom",18));
        students.add(new Student("ml",18));

        //第一種遍歷方式,增強for迴圈
        for (Student student:students){
            System.out.println(student);
        }

        //使用泛型方式給HashMap放入3個學生對象
        HashMap<String, Student> hm = new HashMap<>();
        hm.put("milan",new Student("milan",34));
        hm.put("jack",new Student("jack",31));
        hm.put("tom",new Student("tom",30));

        //2.迭代器
        Set<Map.Entry<String, Student>> entries = hm.entrySet();
        Iterator<Map.Entry<String, Student>> iterator = entries.iterator();
        System.out.println("==========================");
        while (iterator.hasNext()){
            Map.Entry<String, Student> next = iterator.next();
            System.out.println(next.getKey()+"-"+next.getValue());

        }
    }
}
class Student{
    private String name;
    private int age;

    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

泛型使用註意事項

1.interface 介面名{}和class 類名<K,V>中T,K,V只能是引用類型。如下麵:

List<Integer> list = new ArrayList<Integer>();//這樣寫正確
List<int> list = new ArrayList<int>();//這樣寫錯誤

2.在給泛型指定具體類型後,可以傳入該類型或者其子類類型

3.泛型使用形式

//1.第一種方式
List<Integer> list1 = new ArrayList<Integer>();
//2.第二種方式
List<Integer> Iist2 = new ArrayList<>();

3.如果我們這樣寫List list3=new ArrayList();預設給它的泛型是 E就是Object.,等價於ArrayList arrayList = new ArrayList();

自定義泛型

自定義泛型類

語法:

class 類名<T,R....>{//......表示可以有多個泛型
	成員
}

註意事項:

1.普通成員可以使用泛型(屬性、方法)

2.使用泛型的數組,不能初始化

3.靜態方法中不能使用類的泛型

4.泛型類的類型,是在創建對象時確定的(因為創建對象時,需要指定確定類型)

5.如果在創建對象時,沒有指定類型,預設為Object

class Tiger<T,R,M>{
    String name;
    R r;//屬性使用到泛型
    M m;
    T t;
    T[] ts;//因為數組在new時不能確定T的類型,就無法在記憶體開空間

    public Tiger(String name) {
        this.name = name;
    }

    public Tiger(R r, M m, T t) {
        this.r = r;
        this.m = m;
        this.t = t;
    }

    public Tiger(String name, R r, M m, T t) {//構造器使用泛型
        this.name = name;
        this.r = r;
        this.m = m;
        this.t = t;
    }

    //因為靜態是和類相關的,在類載入時,對象還沒有創建
    //所以,如果靜態方法和靜態屬性使用了泛型,JVM就無法完成初始化
    //static R r2;
//    public static void m1(M m){
//
//    }

    public R getR() {//返回類型可以使用泛型
        return r;
    }

    public void setR(R r) {//方法使用到泛型
        this.r = r;
    }
}

自定義泛型介面

語法:

interface 介面名<T,R....>{

}

註意事項:

1.介面中,靜態成員也不能使用泛型(這個和泛型類規定一樣)

2.泛型介面的類型,在繼承介面或者實現介面時確定

3.沒有指定類型,預設為Object

/**
 * 泛型介面使用的說明
 * 1.介面中,靜態成員也不能使用泛型
 * 2.泛型介面的類型,在繼承介面或者實現介面時確定
 * 3.沒有指定類型,預設為Object
 */

//在繼承介面指定泛型介面的類型
interface IA extends IUsb<String,Double>{

}
//當我們去實現IA介面時,因為IA在繼承IUsub介面時,指定了U為String, R為Double,在實現IUsub介面的方法時,使用String替換U,是Double替換R
class AA implements IA{

    @Override
    public Double get(String s) {
        return null;
    }

    @Override
    public void hi(Double aDouble) {

    }

    @Override
    public void run(Double r1, Double r2, String u1, String u2) {

    }
}
//實現介面時,直接指定泛型介面的類型
//給U指定Integer給R指定了Float
//所以,當我們實現IUsb方法時,會使用Integer替換U,使用Float替換R
class BB implements IUsb<Integer,Float>{

    @Override
    public Float get(Integer integer) {
        return null;
    }

    @Override
    public void hi(Float aFloat) {

    }

    @Override
    public void run(Float r1, Float r2, Integer u1, Integer u2) {

    }
}
//沒有指定類型,預設為Object
//建議直接寫成IUsb<Object,Object>
class CC implements IUsb{//等價class CC implements IUsb<Object,,Object>
    @Override
    public Object get(Object o) {
        return null;
    }

    @Override
    public void hi(Object o) {

    }

    @Override
    public void run(Object r1, Object r2, Object u1, Object u2) {

    }

}
interface IUsb<U,R>{
    //普通方法中,可以使用介面泛型
    R get(U u);

    void hi(R r);

    void run(R r1,R r2,U u1,U u2);

    //在jdk8中,可以在介面中,使用預設方法,也是可以使用泛型
    default R method(U u){
        return null;
    }
}

自定義泛型方法

語法

修飾符 <T,R...>返回類型 方法名(參數列表){

}

註意事項:

1.泛型方法,可以定義在普通類中,也可以定義在泛型類中

2.當泛型方法被調用時,類型會確定

3.public void eat(E e){},修飾符後設有<T,R.>eat方法不是泛型方法,而是使用了泛型

public class CustomMethodGeneric {
    public static void main(String[] args) {
        Car car = new Car();
        car.fly("寶馬",100);//當調用方法時,傳入參數,編譯器,就會確定類型
        System.out.println("==================");


        Fish<String, ArrayList> fish = new Fish<>();
        fish.hello(new ArrayList(),11.3f);
    }
}
//泛型方法,可以定義在普通類中,也可以定義在泛型類中
class Car{//普通類
    public void run(){//普通方法

    }

    //泛型方法
    //1.<T,R>就是泛型
    //2.是提供給 fly使用的
    public <T,R> void fly(T t,R r){//泛型方法
        System.out.println(t.getClass());
        System.out.println(r.getClass());
    }
}
class Fish<T,R>{//泛型類

    public <U,M> void eat(U u,M m){//泛型方法

    }

    //下麵的這個hi方法不是泛型方法,是hi方法使用了類聲明的泛型
    public void hi(T t){

    }

    //泛型方法,可以使用類聲明的泛型,也可以使用自己聲明泛型
    public <K> void hello(R r,K k){
        System.out.println(r.getClass());
        System.out.println(k.getClass());
    }
}

泛型的繼承和通配符

說明:

1.泛型不具備繼承性

2.<?>:支持任意泛型類型

3.<? extends A>:支持A類以及A類的子類,規定了泛型的上限

4.<? super A>:支持A類以及A類的父類,不限於直接父類,規定了泛型的下限

public class GenericExtends {
    public static void main(String[] args) {
        //泛型沒有繼承性
        //List<Object> list = new ArrayList<String>();

        List<Object> list1 = new ArrayList<>();
        List<String> list2 = new ArrayList<>();
        List<AA> list3 = new ArrayList<>();
        List<BB> list4 = new ArrayList<>();
        List<CC> list5 = new ArrayList<>();

        //如果是List<?>c,可以接受任意的泛型類型
        printCollection1(list1);
        printCollection1(list2);
        printCollection1(list3);
        printCollection1(list4);
        printCollection1(list5);

        //List<? extends AA>c:表示上限,可以接受AA或者AA子類
        //printCollection2(list1);  error
        //printCollection2(list2);  error
        printCollection2(list3);
        printCollection2(list4);
        printCollection2(list5);

        //List<?super AA>c:支持AA類以及AA類的父類,不限於直接父類
        printCollection3(list1);
        //printCollection3(list2); error
        printCollection3(list3);
        //printCollection3(list4); error
        //printCollection3(list5); error
    }
    //說明:List<?>表示任意的泛型類型都可以接受
    public static void printCollection1(List<?> c){
        for (Object object :c){
            System.out.println(object);
        }
    }

    //?extends AA表示上限,可以接受AA或者AA子類
    public static void printCollection2(List<? extends AA> c){
        for (Object object:c){
            System.out.println(object);
        }
    }

    //?super 子類類名AA:支持AA類以及AA類的父類,不限於直接父類,規定了泛型的下限
    public static void printCollection3(List<? super AA> c){
        for (Object object:c){
            System.out.println(object);
        }
    }
}
class AA{

}
class BB extends AA {

}

class CC extends BB {

}


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

-Advertisement-
Play Games
更多相關文章
  • 在本篇文章中,我們詳細介紹了 Flutter 進階的主題,包括導航和路由、狀態管理、非同步處理、HTTP請求和Rest API,以及數據持久化。這些主題在實際應用中都非常重要,幫助你構建更複雜、功能更強大的 Flutter 應用。 ...
  • 在SpringBoot的Controller中,可以使用註解@RequestBody來獲取POST請求中的JSON數據。我們可以將這個註解應用到一個Controller方法的參數上,Spring將會負責讀取請求正文中的數據,將其反序列化為一個Java對象,並將其作為Controller方法的參數傳遞 ...
  • 對於從事後端開發的同學來說,線程安全問題是我們每天都需要考慮的問題。 線程安全問題通俗地講主要是在多線程的環境下,不同線程同時讀和寫公共資源(臨界資源)導致的數據異常問題。 比如:變數a=0,線程1給該變數+1,線程2也給該變數+1。此時,線程3獲取a的值有可能不是2,而是1。線程3這不就獲取了錯誤 ...
  • ### 歡迎訪問我的GitHub > 這裡分類和彙總了欣宸的全部原創(含配套源碼):[https://github.com/zq2599/blog_demos](https://github.com/zq2599/blog_demos) ### 關於bean的作用域(scope) - 官方資料:ht ...
  • ## 開篇-為什麼要使用線程池? ​ Java 中的線程池是運用場景最多的併發框架,幾乎所有需要非同步或併發執行任務的程式都可以使用線程池。在開發過程中,合理地使用線程池能夠帶來 3 個好處。 ​ 第一:降低資源消耗。通過重覆利用已創建的線程降低線程創建和銷毀造成的消耗。 ​ 第二:提高響應速度。當任 ...
  • ## 實踐環境 Win10 Java JDK1.8 ## 代碼實現 pom.xml配置 ```xml 4.0.0 com.shouke example 1.0 1.8 ${java.version} ${java.version} 4.1.2 org.apache.poi poi-ooxml ${p ...
  • 註冊表具有唯一標識,用於管理多個日誌 ```c++ // Copyright(c) 2015-present, Gabi Melman & spdlog contributors. // Distributed under the MIT License (http://opensource.org ...
  • ## vs中創建Filter 在一個新項目中右鍵 - Add - New,預設只有一選項 New Filter。 創建出來的Filter可以理解為是VS的過濾器(虛擬目錄),它不會在本地的磁碟上新建目錄,而是修改了.filters文件,把這種目錄關係記錄在.filters文件中。 ![image-2 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...