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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈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…