跳到主要内容

Java 泛型详解

Java 泛型(Generics)是 Java 5(JDK 1.5)引入的一个重要特性,可以在编译时检查类型的正确性,增强代码的安全性和可读性。

泛型的使用

泛型只能使用引用类型,不能使用基本类型。泛型的具体类型在使用时才被确定,是一种类型参数化的机制。它提供了编译时的类型安全保障,提高了程序的复用性,而且不需要进行强制类型转换。

package com.generics;

public class Printer<T, K> {
private T firstValue;
private K secondValue;

public Printer(T firstValue, K secondValue) {
this.firstValue = firstValue;
this.secondValue = secondValue;
}

public T print() {
System.out.println(firstValue);
System.out.println(secondValue);
return firstValue;
}

public <V> boolean myEquals(V val) {
return val.equals(firstValue);
}
}
Printer<Integer, String> integerPrinter = new Printer<>(100, "sumingcheng");
integerPrinter.print();

Printer<String, Number> stringPrinter = new Printer<>("sumingcheng", 1);
String result = stringPrinter.print();
System.out.println(result);

boolean isEqual = stringPrinter.myEquals("sumingcheng");
System.out.println(isEqual);

菱形操作符

从 Java 7 开始,可以使用菱形操作符 <> 在实例化对象时进行类型推断。这样不需要在 new 操作符后的构造函数中再次显式地指定泛型类型,编译器会根据变量的类型自动推断。

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

编译器会从左侧的 ArrayList<String> 推断出右侧的泛型类型,使代码更加简洁。

泛型类

使用泛型类时,需要在类名后面使用尖括号定义泛型标识,一般使用 T、E、K、V 等作为标识符。有几个泛型标识,就需要传入几个具体的类型。传入具体类型时,就相当于替换了泛型标识。

如果在实例化时没有传入具体类型,泛型的默认上限是 Object 类型。需要注意的是,泛型不支持基本数据类型,只支持引用类型,因为泛型类型是通过 Object 类进行处理的,而基本数据类型无法继承 Object

public class MyTest<T, E, K, V> {
private T first;
private E second;
private K third;
private V fourth;

public MyTest(T first, E second, K third, V fourth) {
this.first = first;
this.second = second;
this.third = third;
this.fourth = fourth;
}

public void printTypes() {
System.out.println("T 的类型是:" + first.getClass().getName());
System.out.println("E 的类型是:" + second.getClass().getName());
System.out.println("K 的类型是:" + third.getClass().getName());
System.out.println("V 的类型是:" + fourth.getClass().getName());
}
}
MyTest<Integer, String, Double, Character> myTest = new MyTest<>(100, "sumingcheng", 99.9, 'A');
myTest.printTypes();

装箱与拆箱

在实例化泛型类时,可以传入基本数据类型的包装类,例如 IntegerString 等。类在实例化过程中,会根据泛型类的具体类型进行装箱。如果没有指定具体的泛型类型,默认使用 Object 进行装箱。取值时,会自动进行拆箱。

package com.generics;

public class Box<T> {
private T value;

public Box() {
}

public Box(T value) {
this.value = value;
}

public T getValue() {
return value;
}

public void setValue(T value) {
this.value = value;
}
}
package com.generics;

public class BoxTest {
public static void main(String[] args) {
Box boxWithoutType = new Box(); // 不指定泛型类型,默认为 Object
boxWithoutType.setValue("sumingcheng"); // 可以存储任何类型的对象
boxWithoutType.setValue(123); // 也可以存储一个 int

Object value = boxWithoutType.getValue(); // 返回值类型为 Object
System.out.println(value);

Box<Integer> integerBox = new Box<>();
integerBox.setValue(42); // 自动装箱:将 int 转换为 Integer

int intValue = integerBox.getValue(); // 自动拆箱:将 Integer 转换为 int
System.out.println(intValue);
}
}

注意事项

  • 泛型只能使用引用类型,不能使用基本类型。
  • 在使用泛型时,类型参数在编译期会进行类型擦除,因此在运行时无法获取实际的泛型类型。
  • 为了避免类型转换异常,应尽量在实例化泛型类时指定具体的类型参数。