
捕获通配符类型是编译器使用的一种类型,它在一个特定位置代表通配符类型的特定实例的类型。
示例:以具有两个通配符参数的方法为例
void m(Ex<?> e1, Ex<?>e2)。声明的类型
e1和
e2被写入完全相同,
Ex<?>。但是
e1和
e2可能具有不同且不兼容的运行时类型。
即使以相同的方式编写类型,类型检查器也不能认为类型相等。因此,在编译期间,
e1和的类型参数将
e2被赋予特定的类型,并在每个使用它们的地方使用新的类型。这些新类型称为其声明类型的
捕获 。
捕获通配符是未知的,但是是普通的和具体的类型。可以与其他类型相同的方式使用。
可以在JLS中找到对此的技术描述:
5.1.10。捕获转化
让G命名具有n个类型参数A1,…,An并具有相应范围U1,…,Un的通用类型声明(第8.1.2节,第9.1.2节)。
存在从参数化类型G(第4.5节)到参数化类型G的捕获转换,其中对于1≤i≤n:
- 如果Ti是形式为?的通配符类型参数(第4.5.1节),则Si是新鲜类型变量,其上限为Ui [A1:= S1,…,An:=
Sn],下限为空类型(第4.1节)。- …
我们可以将其应用于您的示例:
public Integer accept(RecipientTypeVisitor<?> visitor){ //Error: return visitor.visit(this); //Cannot convert capture of #1 to Integer}RecipientTypeVisitor在编译期间引入了对type参数的捕获。捕获的类型参数是具体的,但完全未知(JLS将此称为“新类型变量”),并且肯定不能转换为
Integer。
无法直接表示捕获通配符类型,因此您不能声明该类型的变量或对其进行过多处理。但是,您可以通过以其为参数调用泛型方法来间接获得其名称。我引用的JLS部分对此有一个很好的示例:
public static void reverse(List<?> list) { rev(list); }private static <T> void rev(List<T> list) { List<T> tmp = new ArrayList<T>(list); for (int i = 0; i < list.size(); i++) { list.set(i, tmp.get(list.size() - i - 1)); }}
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)