Java 泛型详解:类型参数的力量

泛型(Generics)是 Java 类型系统的重要支柱。

它让代码更安全、更简洁、更通用,是理解现代 Java 编程的必修课。

本文将系统讲解 Java 泛型的定义、使用、通配符、边界与常见陷阱。


一、为什么需要泛型

在 Java 1.5 之前,集合只能存放 Object 类型:

java 复制代码
List list = new ArrayList();
list.add("Hello");
list.add(123); // 不安全
String s = (String) list.get(0); // 强制类型转换

缺点:类型不安全、容易出错。

Java 1.5 引入 泛型机制(Generics),允许在定义类或方法时引入"类型参数"。

java 复制代码
List<String> list = new ArrayList<>();
list.add("Hello");
// list.add(123); // 编译错误
String s = list.get(0); // 无需强转

二、泛型的基本语法

java 复制代码
class Box<T> {          // 定义泛型类
    private T value;
    public void set(T value) { this.value = value; }
    public T get() { return value; }
}

Box<Integer> box = new Box<>();
box.set(100);
Integer data = box.get();
关键点 说明
<T> 声明类型参数(Type Parameter)
T 类型变量,可代表任意引用类型
Box<Integer> 实例化泛型类

三、泛型类与泛型接口

泛型类

java 复制代码
class Pair<K, V> {
    private K key;
    private V value;
    public Pair(K key, V value) { this.key = key; this.value = value; }
    public K getKey() { return key; }
    public V getValue() { return value; }
}

泛型接口

java 复制代码
interface Converter<F, T> {
    T convert(F from);
}

class StringToInteger implements Converter<String, Integer> {
    public Integer convert(String from) {
        return Integer.valueOf(from);
    }
}

四、泛型方法

方法也可以定义自己的类型参数:

java 复制代码
public static <T> void printArray(T[] array) {
    for (T element : array) {
        System.out.println(element);
    }
}

String[] arr = {"A", "B", "C"};
printArray(arr); // 自动推断 T = String

⚡ 泛型方法的类型参数与类的泛型参数独立存在。


五、通配符:? 的三种形态

通配符 ? 表示未知类型,常用于方法参数中。

通配符 含义 示例
? 任意类型 List<?> list
? extends T T 或其子类(上界通配符) List<? extends Number>
? super T T 或其父类(下界通配符) List<? super Integer>

示例:

java 复制代码
public static void show(List<? extends Number> list) {
    for (Number n : list) {
        System.out.println(n);
    }
}
写入能力 读取能力
? extends T ✅ 读安全 ❌ 写
? super T ✅ 写安全 ❌ 读(只能读为 Object)

六、边界与多重限制

java 复制代码
class Calculator<T extends Number & Comparable<T>> {
    public boolean isGreater(T a, T b) {
        return a.compareTo(b) > 0;
    }
}

✅ 可指定多个上界,必须先是类后是接口。


七、泛型数组与类型推断

  • ❌ 泛型数组不能直接创建:new T[10](类型擦除原因)
  • ✅ 可使用 List 或 Array.newInstance() 代替。

类型推断:

java 复制代码
List<String> list = List.of("A", "B");
var map = Map.of(1, "A", 2, "B"); // Java 10+

八、泛型的好处与限制

优点 缺点
✅ 编译期类型安全 ❌ 运行时类型擦除
✅ 代码复用性高 ❌ 无法创建泛型数组
✅ 无需强转 ❌ 不能用基本类型

九、常见泛型陷阱

❌ 泛型类型不能用于静态上下文:

java 复制代码
class Box<T> {
    // static T value; // 错误!
}

❌ 泛型类型无法用于 instanceof:

java 复制代码
if (obj instanceof Box<String>) // 错误:类型擦除导致编译不通过

🔚 总结

分类 关键语法 示例
泛型类 <T> class Box<T>
泛型方法 <T> static <T> void print(T t)
通配符 ? extends / ? super List<? super Integer>
边界 T extends Number 限制类型范围
相关推荐
大学生资源网几秒前
基于springboot的万亩助农网站的设计与实现源代码(源码+文档)
java·spring boot·后端·mysql·毕业设计·源码
小严家2 分钟前
Java基础教程大全完整学习路径
java·开发语言·学习
毕设源码-朱学姐3 分钟前
【开题答辩全过程】以 基于Java的电影推荐系统为例,包含答辩的问题和答案
java·开发语言
sheji34168 分钟前
【开题答辩全过程】以 基于SSM的校园新冠疫苗接种信息管理系统为例,包含答辩的问题和答案
java·开发语言
菜鸟233号10 分钟前
力扣78 子集 java实现
java·数据结构·算法·leetcode
dddaidai12319 分钟前
深入JVM(四):垃圾收集器
java·开发语言·jvm
BBB努力学习程序设计25 分钟前
Java方法(函数)完全指南:初学者的第一个"工具箱"
java
爬山算法38 分钟前
Netty(19)Netty的性能优化手段有哪些?
java·后端
love is sour1 小时前
深入浅出 jmap:Java 内存分析的“显微镜“
java·开发语言·测试工具·性能优化
想用offer打牌1 小时前
虚拟内存与寻址方式解析(面试版)
java·后端·面试·系统架构