java泛型类型的调用和实例化

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

Java中的泛型是一种在编译期进行类型检查的机制,它允许你在定义类、接口和方法时使用类型参数,从而提高代码的复用性和安全性。但泛型的调用和实例化有一些限制和注意事项,下面来详细说明。

泛型类的调用

当你定义一个泛型类时,比如:

public class Box {
    private T value;

    public void set(T value) {
        this.value = value;
    }

    public T get() {
        return value;
    }
}

在使用这个类时,你需要指定具体的类型参数:

Box stringBox = new Box();
stringBox.set("Hello");
String content = stringBox.get();

这里的 Box 就是泛型类型的调用,表示这个 Box 实例只能存放 String 类型的对象。

泛型方法的调用

泛型方法是在方法声明中引入类型参数的方法。例如:

public void printArray(E[] array) {
    for (E element : array) {
        System.out.println(element);
    }
}

调用这个方法时,Java通常能根据传入的参数自动推断类型:

String[] words = {"a", "b", "c"};
printArray(words); // 自动推断 E 为 String

你也可以显式指定类型(较少用):

this.printArray(words);

泛型不能直接实例化

由于Java泛型使用的是类型擦除机制,在运行时泛型信息会被擦除,因此你不能直接通过泛型类型参数创建实例:

public class Container {
    // 错误:无法实例化 T
    // T obj = new T(); // 编译错误!
}

如果需要创建泛型类型的实例,有以下几种替代方式:

  • 通过反射传入 Class 对象:
  • public T createInstance(Class clazz) throws Exception {
        return clazz.newInstance();

    }
  • 使用工厂模式或函数式接口:
  • public T create(Supplier supplier) {
        return supplier.get();
    }

泛型数组的限制

你不能直接创建泛型类型的数组:

// 错误:
// Box[] boxes = new Box[10]; // 编译错误

// 合法但不推荐(会警告):
Box>[] boxes = new Box>[10];

因为类型擦除导致运行时无法保证数组元素的类型安全。建议使用集合类如 ArrayList 来代替。

基本上就这些。泛型的重点在于编译期类型安全,而不是运行时创建实例。理解类型擦除和合理使用通配符、边界等特性,才能正确使用泛型。不复杂但容易忽略细节。