Java 高級泛型實戰:8 個場景化編程技巧

文章目錄

    • 一、通配符高級應用:靈活處理類型關系
    • 二、泛型方法與類型推斷
    • 三、泛型類的嵌套使用
    • 四、受限泛型與邊界條件
    • 五、泛型與反射結合
    • 六、泛型在函數式接口中的應用
    • 七、類型擦除與橋接方法
    • 八、自定義泛型注解
    • 總結

在Java編程中,泛型不僅是類型安全的保障,更是提升代碼復用性和靈活性的利器。本文將結合8個典型場景,深入剖析高級泛型的應用技巧,幫助開發者突破基礎用法,掌握泛型在復雜業務中的實戰策略。

一、通配符高級應用:靈活處理類型關系

通配符 ? 用于解決類型間的兼容性問題。<? extends T> 表示類型上限,只能獲取元素;<? super T> 表示類型下限,只能插入元素。以集合操作為例:

import java.util.ArrayList;
import java.util.List;class Animal {}
class Dog extends Animal {}public class Main {public static void main(String[] args) {List<Dog> dogs = new ArrayList<>();// 讀取數據,使用? extendsreadElements(dogs); // 寫入數據,使用? superwriteElement(dogs); }// 只能讀取,不能寫入public static void readElements(List<? extends Animal> list) {for (Animal animal : list) {System.out.println(animal);}}// 只能寫入,不能讀取public static void writeElement(List<? super Dog> list) {list.add(new Dog());}
}

上述代碼中,readElements 方法確保讀取的元素至少是 Animal 類型,writeElement 方法保證寫入的 Dog 元素能被正確接收。

二、泛型方法與類型推斷

泛型方法可以獨立于類定義,通過類型推斷簡化代碼。例如,實現一個通用的交換方法:

public class Main {public static <T> void swap(T[] array, int i, int j) {T temp = array[i];array[i] = array[j];array[j] = temp;}public static void main(String[] args) {Integer[] numbers = {1, 2};swap(numbers, 0, 1);for (Integer num : numbers) {System.out.println(num);}}
}

swap 方法的類型參數 T 由調用時傳入的數組類型自動推斷,無需顯式指定。

三、泛型類的嵌套使用

在復雜的數據結構中,泛型類的嵌套能提供強大的表達能力。以多層容器為例:

import java.util.ArrayList;
import java.util.List;class Outer<T> {private T value;private List<Inner<T>> innerList = new ArrayList<>();public Outer(T value) {this.value = value;}public void addInner(Inner<T> inner) {innerList.add(inner);}static class Inner<T> {private T innerValue;public Inner(T innerValue) {this.innerValue = innerValue;}}
}public class Main {public static void main(String[] args) {Outer<String> outer = new Outer<>("Outer");Outer.Inner<String> inner = outer.new Inner<>("Inner");outer.addInner(inner);}
}

通過嵌套泛型類,Outer 類不僅存儲自身類型的數據,還能管理包含相同類型數據的 Inner 類實例。

四、受限泛型與邊界條件

使用 extends 關鍵字限制泛型類型,要求類型必須實現特定接口或繼承某個類。比如,實現一個計算幾何圖形面積的通用方法:

interface Shape {double getArea();
}class Rectangle implements Shape {private double width;private double height;public Rectangle(double width, double height) {this.width = width;this.height = height;}@Overridepublic double getArea() {return width * height;}
}class Circle implements Shape {private double radius;public Circle(double radius) {this.radius = radius;}@Overridepublic double getArea() {return Math.PI * radius * radius;}
}public class Main {public static <T extends Shape> double totalArea(List<T> shapes) {double sum = 0;for (T shape : shapes) {sum += shape.getArea();}return sum;}public static void main(String[] args) {List<Shape> shapeList = new ArrayList<>();shapeList.add(new Rectangle(3, 4));shapeList.add(new Circle(5));System.out.println(totalArea(shapeList));}
}

totalArea 方法限定 T 必須是 Shape 接口的實現類,確保傳入的對象都具備計算面積的能力。

五、泛型與反射結合

利用反射可以在運行時獲取泛型的實際類型。在數據反序列化場景中,這一特性尤為重要:

import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.List;class GenericList<T> {private List<T> list = new ArrayList<>();public void add(T element) {list.add(element);}public Type getActualTypeArgument() {Type genericSuperclass = getClass().getGenericSuperclass();if (genericSuperclass instanceof ParameterizedType) {ParameterizedType parameterizedType = (ParameterizedType) genericSuperclass;return parameterizedType.getActualTypeArguments()[0];}return null;}
}public class Main {public static void main(String[] args) {GenericList<String> stringList = new GenericList<>();stringList.add("Hello");Type type = stringList.getActualTypeArgument();System.out.println("實際類型: " + type.getTypeName());}
}

通過反射獲取 ParameterizedType,可解析出泛型類實例化時的具體類型。

六、泛型在函數式接口中的應用

Java 8的函數式接口結合泛型,能實現更靈活的操作。以 Function 接口為例:

import java.util.function.Function;class Person {private String name;private int age;public Person(String name, int age) {this.name = name;this.age = age;}public String getName() {return name;}public int getAge() {return age;}
}public class Main {public static void main(String[] args) {Function<Person, String> getName = Person::getName;Person person = new Person("Alice", 25);System.out.println(getName.apply(person));}
}

Function 接口的泛型參數定義了輸入和輸出類型,通過方法引用實現類型安全的操作。

七、類型擦除與橋接方法

泛型在編譯后會發生類型擦除,這可能導致一些意想不到的問題。例如,子類覆蓋父類泛型方法時,編譯器會生成橋接方法:

class GenericParent<T> {public void method(T t) {System.out.println("Parent method: " + t);}
}class GenericChild extends GenericParent<String> {@Overridepublic void method(String s) {System.out.println("Child method: " + s);}
}public class Main {public static void main(String[] args) {GenericChild child = new GenericChild();child.method("Hello");}
}

理解類型擦除和橋接方法的原理,有助于處理泛型繼承和覆蓋中的兼容性問題。

八、自定義泛型注解

結合泛型與注解,可以實現更強大的元編程能力。例如,定義一個用于驗證數據類型的注解:

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
@interface ValidateType<T> {Class<T> value();
}class Validator {@ValidateType(String.class)public static void validate(Object obj) {if (!(obj instanceof String)) {throw new IllegalArgumentException("類型不匹配");}System.out.println("驗證通過");}
}public class Main {public static void main(String[] args) {Validator.validate("Test");}
}

自定義泛型注解能在運行時根據具體類型進行動態驗證,提升代碼的健壯性和可維護性。

總結

Java高級泛型通過通配符、受限類型、反射結合等技巧,在提升代碼類型安全、復用性和靈活性方面發揮著關鍵作用。從集合操作時通配符對類型兼容性的把控,到函數式接口與泛型結合實現的靈活操作;從反射獲取泛型實際類型解決反序列化難題,到自定義泛型注解實現動態驗證 ,這些技巧貫穿于數據結構設計、算法實現、框架開發等多個場景。同時,理解類型擦除和橋接方法的原理,能幫助開發者規避泛型使用中的潛在問題。在實際開發中,合理運用這些技巧,不僅能編寫出更簡潔、高效的代碼,還能增強系統的可擴展性與穩定性,讓Java編程更具專業性與規范性。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/86389.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/86389.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/86389.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

[藍橋杯 2024 國 B] 立定跳遠

問題描述 在運動會上&#xff0c;小明從數軸的原點開始向正方向立定跳遠。項目設置了 n 個檢查點 a1,a2,...,an且 ai≥ai?1>0。小明必須先后跳躍到每個檢查點上且只能跳躍到檢查點上。同時&#xff0c;小明可以自行再增加 m 個檢查點讓自己跳得更輕松。在運動會前&#xf…

2025年全國I卷數學壓軸題解答

第19題第3問: b b b 使得存在 t t t, 對于任意的 x x x, 5 cos ? x ? cos ? ( 5 x t ) < b 5\cos x-\cos(5xt)<b 5cosx?cos(5xt)<b, 求 b b b 的最小值. 解: b b b 的最小值 b m i n min ? t max ? x g ( x , t ) b_{min}\min_{t} \max_{x} g(x,t) bmi…

wpf在image控件上快速顯示內存圖像

wpf在image控件上快速顯示內存圖像https://www.cnblogs.com/haodafeng/p/10431387.html 如果你在尋找能夠快速在image控件刷新大圖像&#xff08;比如分辨率3000*3000的圖像&#xff09;的辦法&#xff0c;尤其是想把內存中的裸數據&#xff08;只有圖像的數據&#xff0c;不包…

解決網頁導出PDF部分內容被遮擋問題

問題描述 以學習通為例&#xff0c;在使用CtrlP打印頁面或截圖時&#xff0c;固定側邊欄會遮擋部分內容&#xff0c;影響完整內容的獲取。如下圖所示&#xff1a; 解決辦法 通過瀏覽器開發者工具臨時移除固定側邊欄&#xff0c;具體步驟如下&#xff1a; 在目標頁面右鍵點…

機器學習監督學習實戰六:五種算法對新聞組英文文檔進行文本分類(20類),詞頻統計和TF-IDF 轉換特征提取方法理論和對比解析

本文主要介紹了20 Newsgroups數據集及其在文本分類任務中的應用。20 Newsgroups數據集包含約20,000篇新聞組文檔&#xff0c;分為20個不同主題的新聞組&#xff0c;數據集被分為訓練集和測試集。在數據預處理階段&#xff0c;使用了CountVectorizer和TfidfVectorizer兩種方法將…

易學探索助手-個人記錄(十四)

項目背景 在大語言模型&#xff08;LLM&#xff09;完成指令微調&#xff08;SFT&#xff09;之后&#xff0c;雖然可以處理開放式問答任務&#xff0c;但在專業領域&#xff08;如《周易》&#xff09;仍面臨知識更新滯后、事實性薄弱等問題。為此&#xff0c;本文介紹如何通…

從“人找政策”到“政策找人”:智能退稅ERP數字化重構外貿生態

離境退稅新政核心內容與外貿企業影響 &#xff08;一&#xff09;政策核心變化解析 退稅商店網絡擴容 新政明確鼓勵在大型商圈、旅游景區、交通樞紐等境外旅客聚集地增設退稅商店&#xff0c;并放寬備案條件至納稅信用M級企業。以上海為例&#xff0c;靜安區計劃新增1000家退…

Pandas 可視化集成:數據科學家的高效繪圖指南

為什么選擇 Pandas 進行數據可視化&#xff1f; 在數據科學和分析領域&#xff0c;可視化是理解數據、發現模式和傳達見解的關鍵步驟。Python 生態系統提供了多種可視化工具&#xff0c;如 Matplotlib、Seaborn、Plotly 等&#xff0c;但 Pandas 內置的可視化功能因其與數據結…

曼昆《經濟學原理》第九版 第十一章公共物品與公共資源

一、物品分類的基本框架 排他性&#xff1a;能否阻止他人使用該物品的特性競爭性&#xff1a;一個人使用是否減少他人使用的特性 根據這兩個特性可將物品分為四類&#xff1a; 私人物品&#xff1a;既有排他性又有競爭性&#xff08;如冰淇淋、衣服&#xff09;公共物品&…

基于大模型預測原發性急性閉角型青光眼的技術方案研究大綱

目錄 一、引言二、技術方案概述三、術前階段(一)數據采集與處理(二)大模型預測(三)手術方案制定(四)麻醉方案確定(五)術前健康教育四、術中階段(一)實時數據監測與輸入(二)手術策略動態調整(三)并發癥預警與處理(四)術中健康教育五、術后階段(一)恢復監測與…

基于React 的 AntD 庫進行前端開發過程中的問題匯總

背景 最近寫了半個月的 React 前端&#xff0c;三年沒寫過 React 前端了&#xff0c;有些生疏了&#xff0c;匯總一下 基于React 前端的 antD 庫編寫過程中的低級問題吧。 PS 一下&#xff0c;半個月沒有發布博客了&#xff0c;C站產品經理又悄默默地改了樣式&#xff0c;博客…

Spring @Scheduled vs XXL-JOB vs DolphinScheduler vs Airflow:任務調度框架全景對比

引言 從單機定時任務到分布式工作流調度&#xff0c;不同場景需要選擇匹配的調度框架。 本文對比 Spring Scheduled、XXL-JOB、DolphinScheduler &#xff08;海豚調度器&#xff09;和 Apache Airflow 的核心差異&#xff0c;助你避免過度設計或功能不足。 一、核心定位與適用…

springMVC-10驗證及國際化

驗證 概述 ● 概述 1. 對輸入的數據(比如表單數據)&#xff0c;進行必要的驗證&#xff0c;并給出相應的提示信息。 2. 對于驗證表單數據&#xff0c;springMVC提供了很多實用的注解, 這些注解由JSR303 驗證框架提供. ●JSR 303 驗證框架 1. JSR 303 的含義 JSR&#xff0…

OpenCV 滑動條調整圖像對比度和亮度

一、知識點 1、int createTrackbar(const String & trackbarname, const String & winname, int * value, int count, TrackbarCallback onChange 0, void * userdata 0); (1)、創建一個滑動條并將其附在指定窗口上。 (2)、參數說明: trackbarname: 創建的…

ReadWriteLock(讀寫鎖)和 StampedLock

1. ReadWriteLock&#xff08;讀寫鎖&#xff09;&#xff1a;實現高性能緩存 總結&#xff1a; 要點 內容 適用場景 讀多寫少、高并發讀取場景&#xff08;如緩存&#xff09; 鎖類型 ReadWriteLock接口&#xff0c;ReentrantReadWriteLock實現 讀鎖 vs 寫鎖 多線程可…

【決勝公務員考試】求職OMG——見面課測驗1

2025最新版&#xff01;&#xff01;&#xff01;6.8截至答題&#xff0c;大家注意呀&#xff01; 博主碼字不易點個關注吧,祝期末順利~~ 1.單選題(2分) 下列說法錯誤的是:&#xff08; B &#xff09; A.選調生屬于公務員系統 B.公務員屬于事業編 C.選調生有基層鍛煉的要求 D…

vue3 el-button 自定義本地圖標

設置不生效的原因可能有&#xff1a;1.style標簽里沒加scoped <style scoped></style>2.本地圖片路徑指向錯誤3.自定義圖片長寬沒設置4.deep深度選擇器使用錯誤&#xff0c;vue3用:deep() <el-tooltip content"重新匹配" placement"top"&g…

如何在最短時間內提升打ctf(web)的水平?

剛剛刷完2遍 bugku 的 web 題&#xff0c;前來答題。 每個人對刷題理解是不同&#xff0c;有的人是看了writeup就等于刷了&#xff0c;有的人是收藏了writeup就等于刷了&#xff0c;有的人是跟著writeup做了一遍就等于刷了&#xff0c;還有的人是獨立思考做了一遍就等于刷了。…

6.8 note

paxos算法_初步感知 Paxos算法保證一致性主要通過以下幾個關鍵步驟和機制&#xff1a; 準備階段 - 提議者向所有接受者發送準備請求&#xff0c;請求中包含一個唯一的編號。 - 接受者收到請求后&#xff0c;會檢查編號&#xff0c;如果編號比它之前見過的都大&#xff0c;就會承…

c++ openssl 使用 DES(數據加密標準)進行加密和解密的基本操作

使用 DES&#xff08;數據加密標準&#xff09;進行加密和解密的基本操作&#xff0c;重點展示了 ECB 和 CBC 模式&#xff0c;并且通過篡改密文的方式來進行攻擊。下面是對每個部分的詳細解析。 1. 結構體 Slip struct Slip {char from[16] { 0 }; // 交易的發起者&#x…