java泛型的类型擦除是指在编译时移除泛型类型信息,替换为原始类型,以保持与旧版本的兼容性。1. 类型擦除意味着list
泛型在Java中主要用于提供编译时类型安全,减少强制类型转换,并允许编写可以应用于多种类型的通用代码。它们的核心作用是参数化类型,使得类、接口和方法可以操作不同类型的对象,而无需为每种类型编写不同的代码。
类型安全、代码重用、性能提升。
Java泛型的一个关键特性是类型擦除。这意味着在编译时,泛型类型信息会被移除,替换为它们的原始类型(raw type)。例如,List
立即学习“Java免费学习笔记(深入)”;
类型擦除的主要原因是为了保持与旧版本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
泛型通配符 ? 用于表示未知类型。它主要用于以下几种场景:
读取泛型集合中的元素: 当你只关心从泛型集合中读取元素,而不关心元素的具体类型时,可以使用 ? 通配符。例如,List> 表示一个包含未知类型元素的列表。
作为方法参数: 当方法需要接受一个泛型类型的参数,但方法内部并不依赖于具体的类型时,可以使用 ? 通配符。例如,void printList(List> list) 方法可以接受任何类型的列表。
上界通配符和下界通配符: 可以使用 ? 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
以 HashMap
// 使用泛型的 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();
以上就是java中的generics关键字作用 泛型generics的3个典型应用的详细内容,更多请关注php中文网其它相关文章!
java怎么学习?java怎么入门?java在哪学?java怎么学才快?不用担心,这里为大家提供了java速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号