java中的generics关键字作用 泛型generics的3个典型应用

Java泛型的类型擦除是指在编译时移除泛型类型信息,替换为原始类型,以保持与旧版本的兼容性。1. 类型擦除意味着list和list在运行时都变为list;2. 其目的是确保java 5引入泛型后仍能兼容之前已有的非泛型代码;3. 类型擦除带来的限制包括无法使用instanceof检查泛型类型、不能创建泛型数组;4. 尽管类型信息被擦除,但编译器仍会在编译期间进行类型检查以保证类型安全。

java中的generics关键字作用 泛型generics的3个典型应用

泛型在Java中主要用于提供编译时类型安全,减少强制类型转换,并允许编写可以应用于多种类型的通用代码。它们的核心作用是参数化类型,使得类、接口和方法可以操作不同类型的对象,而无需为每种类型编写不同的代码。

java中的generics关键字作用 泛型generics的3个典型应用

类型安全、代码重用、性能提升。

java中的generics关键字作用 泛型generics的3个典型应用

如何理解Java泛型的类型擦除?

Java泛型的一个关键特性是类型擦除。这意味着在编译时,泛型类型信息会被移除,替换为它们的原始类型(raw type)。例如,List 在运行时会变为 List。

立即学习Java免费学习笔记(深入)”;

java中的generics关键字作用 泛型generics的3个典型应用

类型擦除的主要原因是为了保持与旧版本Java的兼容性。在Java 5引入泛型之前,已经存在大量的Java代码,如果泛型在运行时保留类型信息,那么这些旧代码将无法与使用泛型的新代码兼容。

尽管类型擦除带来了兼容性,但也带来了一些限制。例如,不能在运行时使用 instanceof 操作符来检查泛型类型,也不能创建泛型数组(例如 new List[])。

List<String> stringList = new ArrayList<>(); stringList.add("Hello");  List<Integer> integerList = new ArrayList<>(); integerList.add(123);  System.out.println(stringList.getClass() == integerList.getClass()); // 输出 true,因为类型擦除

泛型方法与普通方法的区别是什么?

泛型方法是指在方法声明中使用了类型参数的方法。类型参数可以用于方法的参数类型、返回类型或方法体中的局部变量类型。与普通方法相比,泛型方法的主要区别在于其类型参数可以在调用方法时指定,从而使方法能够处理不同类型的对象。

普通方法在声明时必须指定参数类型和返回类型,而泛型方法可以根据调用时传入的参数类型来推断类型参数。这使得泛型方法更加灵活和通用。

// 泛型方法示例 public static <T> T findMax(T[] array) {     if (array == null || array.length == 0) {         return null;     }      T max = array[0];     for (int i = 1; i < array.length; i++) {         if (((Comparable<T>) array[i]).compareTo(max) > 0) {             max = array[i];         }     }     return max; }  // 调用泛型方法 Integer[] intArray = {1, 5, 3, 8, 2}; Integer maxInt = findMax(intArray); // 类型参数 T 被推断为 Integer  String[] strArray = {"apple", "banana", "orange"}; String maxStr = findMax(strArray); // 类型参数 T 被推断为 String

泛型通配符 ? 的使用场景有哪些?

泛型通配符 ? 用于表示未知类型。它主要用于以下几种场景:

  1. 读取泛型集合中的元素: 当你只关心从泛型集合中读取元素,而不关心元素的具体类型时,可以使用 ? 通配符。例如,List> 表示一个包含未知类型元素的列表。

  2. 作为方法参数: 当方法需要接受一个泛型类型的参数,但方法内部并不依赖于具体的类型时,可以使用 ? 通配符。例如,void printList(List> list) 方法可以接受任何类型的列表。

  3. 上界通配符和下界通配符: 可以使用 ? extends Type 表示类型参数必须是 Type 或其子类,使用 ? super Type 表示类型参数必须是 Type 或其父类

// 使用通配符读取泛型集合 public static void printList(List<?> list) {     for (Object element : list) {         System.out.println(element);     } }  // 使用上界通配符 public static <T extends Number> double sum(List<T> list) {     double sum = 0;     for (Number number : list) {         sum += number.doubleValue();     }     return sum; }  // 使用下界通配符 public static void addIntegers(List<? super Integer> list) {     list.add(1);     list.add(2); }

泛型在集合框架中的具体应用案例?

Java集合框架广泛使用了泛型,例如 ArrayList、LinkedList、HashMap 等。泛型的使用使得集合框架能够存储和操作各种类型的对象,同时保证类型安全。

以 HashMap 为例,K 表示键的类型,V 表示值的类型。通过指定键和值的类型,可以在编译时检查类型错误,避免在运行时出现 ClassCastException。

// 使用泛型的 HashMap HashMap<String, Integer> ageMap = new HashMap<>(); ageMap.put("Alice", 30); ageMap.put("Bob", 25);  int aliceAge = ageMap.get("Alice"); // 不需要强制类型转换 System.out.println("Alice's age: " + aliceAge);

如何自定义泛型类和泛型接口?

自定义泛型类和泛型接口允许你创建可以操作不同类型的通用类和接口。在类或接口的声明中使用类型参数,并在类或接口的成员中使用这些类型参数。

// 自定义泛型类 class Box<T> {     private T value;      public Box(T value) {         this.value = value;     }      public T getValue() {         return value;     }      public void setValue(T value) {         this.value = value;     } }  // 自定义泛型接口 interface GenericInterface<T> {     T process(T input); }  // 实现泛型接口 class StringProcessor implements GenericInterface<String> {     @Override     public String process(String input) {         return input.toUpperCase();     } }  // 使用自定义泛型类 Box<Integer> integerBox = new Box<>(10); int intValue = integerBox.getValue();  Box<String> stringBox = new Box<>("Hello"); String stringValue = stringBox.getValue();

© 版权声明
THE END
喜欢就支持一下吧
点赞15 分享