Java泛型方法中什么是下界通配符?


泛型是Java中一个概念,您可以通过它使类、接口和方法能够接受所有(引用)类型作为参数。换句话说,它是一个允许用户动态选择方法、类构造函数接受的引用类型的概念。通过将类定义为泛型,您使其类型安全,即它可以作用于任何数据类型。

要定义泛型类,您需要在类名后面的尖括号“<>”中指定要使用的类型参数,您可以将其视为实例变量的数据类型并继续编写代码。

示例

 在线演示

class Student<T>{
   T age;
   Student(T age){
      this.age = age;
   }
   public void display() {
      System.out.println("Value: "+this.age);
   }
}
public class GenericsExample {
   public static void main(String args[]) {
      Student<Float> std1 = new Student<Float>(25.5f);
      std1.display();
      Student<String> std2 = new Student<String>("25");
      std2.display();
      Student<Integer> std3 = new Student<Integer>(25);
      std3.display();
   }
}

输出

Value: 25.5
Value: 25
Value: 25

通配符

除了泛型中的类型参数(T)之外,您还可以使用“?”来表示未知类型。您可以将通配符用作:

  • 参数类型。
  • 字段。
  • 局部字段。

通配符的唯一限制是,在调用泛型方法时,不能将其用作泛型方法的类型参数。

Java 提供了三种类型的通配符,即上界通配符、下界通配符和无界通配符。

下界通配符

上界通配符允许将特定类的所有子类型用作类型参数。

类似地,如果我们使用下界通配符,则可以将“?”的类型限制为特定类型或其超类型。

例如,如果要将 Collection 对象作为方法的参数,并且类型参数为 Integer 类的超类,则只需声明一个以 Integer 类作为下界的通配符。

要创建/声明下界通配符,只需在“?”后指定 super 关键字,然后是类名。

示例

以下 Java 示例演示了如何创建下界通配符。

 在线演示

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Iterator;
public class LowerBoundExample {
   public static void sampleMethod(Collection<? super Integer> col){
      Iterator it = col.iterator();
      while (it.hasNext()) {
         System.out.print(it.next()+" ");
      }
      System.out.println("");
   }
   public static void main(String args[]) {
      ArrayList<Integer> col1 = new ArrayList<Integer>();
      col1.add(24);
      col1.add(56);
      col1.add(89);
      col1.add(75);
      col1.add(36);
      sampleMethod(col1);
      List<Object> col2 = Arrays.asList(22.1f, 3.32f, 51.4f, 82.7f, 95.4f, 625.f);
      sampleMethod(col2);
   }
}

输出

24 56 89 75 36
22.1 3.32 51.4 82.7 95.4 625.0

如果将类型不是 Integer 及其超类型的 Collection 对象作为上述程序中 sampleMethod() 的参数传递,则会生成编译时错误。

示例 

在线演示

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Iterator;
import java.util.HashSet;
public class LowerBoundExample {
   public static void sampleMethod(Collection<? super Integer> col){
      Iterator it = col.iterator();
      while (it.hasNext()) {
         System.out.print(it.next()+" ");
      }
      System.out.println("");
   }
   public static void main(String args[]) {
      ArrayList<Integer> col1 = new ArrayList<Integer>();
      col1.add(24);
      col1.add(56);
      col1.add(89);
      col1.add(75);
      col1.add(36);
      sampleMethod(col1);
      List<Object> col2 = Arrays.asList(22.1f, 3.32f, 51.4f, 82.7f, 95.4f, 625.f);
      sampleMethod(col2);
      HashSet<Double> col3 = new HashSet<Double>();
      col3.add(25.225d);
      col3.add(554.32d);
      col3.add(2254.22d);
      col3.add(445.21d);
      sampleMethod(col3);
   }
}

编译时错误

LowerBoundExample.java:34: error: incompatible types: HashSet<Double> cannot be converted to Collection<? super Integer>
      sampleMethod(col3);
                  ^
Note: Some messages have been simplified; recompile with -Xdiags:verbose to get full output
1 error

更新于: 2019年9月12日

271 次浏览

开启您的 职业生涯

通过完成课程获得认证

开始学习
广告