在 Java 的泛型类型中使用通配符
2005-09-17 10:55
567 查看
在 Java 的泛型类型中使用通配符
Java 从版本5起开始引入泛型(generics)机制。我们知道,Java 的泛型类型如同 java.lang.String,java.io.File 一样,属于普通的 Java 类型。比方说,下面两个变量的类型就是互不相同的:
List<Object>
listObj = new ArrayList<Object>();
List<String> listStr = new ArrayList<String>();
虽然 String 是 Object 的子类,但是 List<String> 和 List<Object> 之间并没有什么关系——List<String> 不是 List<Object> 的子类或者子类型。在下面的代码中,我们会看到将具有 List<Object> 类型的变量赋给期望 List<Object> 类型参数的方法的话,编译器在编译期将会报告一个编译错误。
import
java.util.ArrayList;
import java.util.List;
public class GenericsTypeTest {
public static void testMtd(List<Object> l) {
}
public static void main(String[]
args) {
List<String> testList = new
ArrayList<String>;
// above line will cause a
compile error
testMtd(testList);
}
}
如果我们希望 testMtd 能够接受任意泛型类型的参数,那么我们应该使用 ? 通配符来满足这个要求。List<?> 任意类型的对象的数组这么一个泛型的类型。我们可以把以上代码改成:
public
static void testMtd(List<?> l) {
}
但是这种情况下,testMtd 的参数可以接受类型可能对于程序员设计的意图而言太广泛了一点。因为我们可能只是希望 testMtd 可以接受 AbstractList 及其子类的类型的变量,而不接受 AbstractSet 甚至 Random、Locale 等类型的变量。我们要对通配符有所限制。幸运的是,Java 5 的泛型机制已经考虑到了这一点,我们可以使用边界通配符(bounded
wildcard)形式来满足这个要求。我们将 testMtd 再修改一下:
Public
static void testMtd(List<? Extends AbstractList>) {
}
这样,List<AbstractList>、List<LinkedList>
等等类型的变量就可以传给 testMtd 方法,而储存其他类型元素的 List 的泛型类型变量传给
testMtd 方法将是非法的。除了上边界通配符(upper bounded wildcard)以外,我们还可以使用下边界通配符(lower bounded wildcard),例如 List<?
super AbstractList>。
最后总结一下使用通配符的泛型类型的三种形式:
GenericType<?>
GenericType<?
extends upperBoundType>
GenericType<?
super lowerBoundType>
Java 从版本5起开始引入泛型(generics)机制。我们知道,Java 的泛型类型如同 java.lang.String,java.io.File 一样,属于普通的 Java 类型。比方说,下面两个变量的类型就是互不相同的:
List<Object>
listObj = new ArrayList<Object>();
List<String> listStr = new ArrayList<String>();
虽然 String 是 Object 的子类,但是 List<String> 和 List<Object> 之间并没有什么关系——List<String> 不是 List<Object> 的子类或者子类型。在下面的代码中,我们会看到将具有 List<Object> 类型的变量赋给期望 List<Object> 类型参数的方法的话,编译器在编译期将会报告一个编译错误。
import
java.util.ArrayList;
import java.util.List;
public class GenericsTypeTest {
public static void testMtd(List<Object> l) {
}
public static void main(String[]
args) {
List<String> testList = new
ArrayList<String>;
// above line will cause a
compile error
testMtd(testList);
}
}
如果我们希望 testMtd 能够接受任意泛型类型的参数,那么我们应该使用 ? 通配符来满足这个要求。List<?> 任意类型的对象的数组这么一个泛型的类型。我们可以把以上代码改成:
public
static void testMtd(List<?> l) {
}
但是这种情况下,testMtd 的参数可以接受类型可能对于程序员设计的意图而言太广泛了一点。因为我们可能只是希望 testMtd 可以接受 AbstractList 及其子类的类型的变量,而不接受 AbstractSet 甚至 Random、Locale 等类型的变量。我们要对通配符有所限制。幸运的是,Java 5 的泛型机制已经考虑到了这一点,我们可以使用边界通配符(bounded
wildcard)形式来满足这个要求。我们将 testMtd 再修改一下:
Public
static void testMtd(List<? Extends AbstractList>) {
}
这样,List<AbstractList>、List<LinkedList>
等等类型的变量就可以传给 testMtd 方法,而储存其他类型元素的 List 的泛型类型变量传给
testMtd 方法将是非法的。除了上边界通配符(upper bounded wildcard)以外,我们还可以使用下边界通配符(lower bounded wildcard),例如 List<?
super AbstractList>。
最后总结一下使用通配符的泛型类型的三种形式:
GenericType<?>
GenericType<?
extends upperBoundType>
GenericType<?
super lowerBoundType>
相关文章推荐
- 在 Java 的泛型类型中使用通配符
- 在 Java 的泛型类型中使用通配符
- Java基础:泛型类型的子类及通配符的使用
- Java基础之泛型——使用通配符类型参数(TryWildCard)
- 在 Java 的泛型类型中使用通配符
- Java基础(21):泛型—泛型的定义、用法与类型通配符的使用方式
- java 泛型之不要使用原生态类型
- Java 泛型总结(三):通配符的使用
- Java 理论与实践: 使用通配符简化泛型使用
- [转贴] Java 理论与实践: 使用通配符简化泛型使用
- Java 理论与实践: 使用通配符简化泛型使用
- [改善Java代码]不同的场景使用不同的泛型通配符
- Java泛型入参的三种通配符使用
- JAVA之旅(二十一)——泛型的概述以及使用,泛型类,泛型方法,静态泛型方法,泛型接口,泛型限定,通配符
- Java 理论与实践: 使用通配符简化泛型使用
- java_泛型,设置类型通配符的上限
- java中泛型学习1之类型通配符(?)
- Java,泛型类型通配符和C#对照
- JAVA之旅(二十一)——泛型的概述以及使用,泛型类,泛型方法,静态泛型方法,泛型接口,泛型限定,通配符
- 黑马程序员--Java基础加强--15.利用反射操作泛型IV【通过反射Method解析泛型方法思路】【通过Method对四种Type子接口类型进行解剖】【使用递归对任意复合泛型类型进行彻底解剖】【个人