泛型在编译期提供类型安全,通过类型参数提升代码复用性;定义泛型类如Box<T>后需指定具体类型如Box<String>来调用;泛型方法可自动推断类型,也可显式指定;由于类型擦除,无法直接实例化T,需借助Class对象或Supplier工厂;禁止创建泛型数组如new Box<String>[],建议用ArrayList替代;核心是编译期检查与合理使用通配符、边界。

Java中的泛型是一种在编译期进行类型检查的机制,它允许你在定义类、接口和方法时使用类型参数,从而提高代码的复用性和安全性。但泛型的调用和实例化有一些限制和注意事项,下面来详细说明。
泛型类的调用
当你定义一个泛型类时,比如:
public class Box<T> {private T value;
public void set(T value) {
this.value = value;
}
public T get() {
return value;
}
}
在使用这个类时,你需要指定具体的类型参数:
Box<String> stringBox = new Box<>();stringBox.set("Hello");
String content = stringBox.get();
这里的 Box<String> 就是泛型类型的调用,表示这个 Box 实例只能存放 String 类型的对象。
立即学习“Java免费学习笔记(深入)”;
泛型方法的调用
泛型方法是在方法声明中引入类型参数的方法。例如:
public <E> void printArray(E[] array) {for (E element : array) {
System.out.println(element);
}
}
调用这个方法时,Java通常能根据传入的参数自动推断类型:
String[] words = {"a", "b", "c"};printArray(words); // 自动推断 E 为 String
你也可以显式指定类型(较少用):
this.<String>printArray(words);泛型不能直接实例化
由于Java泛型使用的是类型擦除机制,在运行时泛型信息会被擦除,因此你不能直接通过泛型类型参数创建实例:
public class Container<T> {// 错误:无法实例化 T
// T obj = new T(); // 编译错误!
}
如果需要创建泛型类型的实例,有以下几种替代方式:
- 通过反射传入 Class 对象: public <T> T createInstance(Class<T> clazz) throws Exception {
- 使用工厂模式或函数式接口: public <T> T create(Supplier<T> supplier) {
return clazz.newInstance();
}
return supplier.get();
}
泛型数组的限制
你不能直接创建泛型类型的数组:
// 错误:// Box<String>[] boxes = new Box<String>[10]; // 编译错误
// 合法但不推荐(会警告):
Box<?>[] boxes = new Box<?>[10];
因为类型擦除导致运行时无法保证数组元素的类型安全。建议使用集合类如 ArrayList 来代替。
基本上就这些。泛型的重点在于编译期类型安全,而不是运行时创建实例。理解类型擦除和合理使用通配符、边界等特性,才能正确使用泛型。不复杂但容易忽略细节。











