ArrayList list = new ArrayList();
list.add(new String("test string"));
list.add(new Integer(9)); // purposely placed here to create a runtime ClassCastException
inspectCollection(list);
}
protected void inspectCollection(Collection aCollection) {
Iterator i = aCollection.iterator();
while (i.hasNext()) {
String element = (String) i.next();
}
}
protected void collectionsExample() {
ArrayList list = new ArrayList();
list.add(new String("test string"));
list.add(new Integer(9)); // purposely placed here to create a runtime ClassCastException
inspectCollection(list);
}
protected void inspectCollection(Collection aCollection) {
Iterator i = aCollection.iterator();
while (i.hasNext()) {
String element = (String) i.next();
}
}
以上的样例程序包含的两个方法,collectionExample方法建立了一个简单的集合类型ArrayList,并在ArrayList中增加了一个String和一个Integer对象.而在inspecCollection方法中,我们迭代这个ArrayList用String进行Cast。我们看第二个方法,就出现了一个问题,Collection在内部用的是Object,而我们要取出Collection中的对象时,需要进行Cast,那么开发者必需用实际的类型进行Cast,像这种向下造型,编译器无
法进行检查,如此一来我们就要冒在代码在运行抛出ClassCastException的危险。我们看inspecCollection方法,编译时没有问题,但在运行时就会抛出ClassCastException异常。所以我们一定要远离这个重大的运行时错误
2.使用Generics
从上一章节中的CassCastException这种异常,我们期望在代码编译时就能够捕捉到,下面我们使用范型修改上一章的样例程序。
[java]
protected void collectionsExample() {
ArrayList
list.add(new String("test string"));
// list.add(new Integer(9)); this no longer compiles
inspectCollection(list);
}
protected void inspectCollection(Collection
Iterator
while(i.hasNext()) {
String element = i.next();
}
}
protected void collectionsExample() {
ArrayList
list.add(new String("test string"));
// list.add(new Integer(9)); this no longer compiles
inspectCollection(list);
}
protected void inspectCollection(Collection
Iterator
while(i.hasNext()) {
String element = i.next();
}
}
从上面第2行我们在创建ArrayList时使用了新语法,在JDK1.5中所有的Collection都加入了Generics的声明。例:
[java]
public class ArrayList
// details omitted...
public void add(E element) {
// details omitted
}
public Iterator
// details omitted
}
}
public class ArrayList
// details omitted...
public void add(E element) {
// details omitted
}
public Iterator
// details omitted
}
}
这个E是一个类型变量,并没有对它进行具体类型的定义,它只是在定义ArrayList时的类型占位符,在Example 2中的我们在定义ArrayList的实例时用String绑定在E上,当我们用add(E element)方法向ArrayList中增加对象时,那么就像下面的写法一样: public void add(String element);因为在ArrayList所有方法都会用String来替代E,无论是方法的参数还是返回值。这时我们在看Example 2中的第四行,编译就会反映出编译错误。
所以在java中增加Generics主要的目的是为了增加类型安全。
通过上面的简单的例子我们看到使用Generics的好处有:
· 1.在类型没有变化时,Collection是类型安全的。
· 2.内在的类型转换优于在外部的人工造型。
· 3.使Java接口更加强壮,因为它增加了类型。
· 4.类型的匹配错误在编译阶段就可以捕捉到,而不是在代码运行时。
受约束类型变量
虽然许多Class被设计成Generics,但类型变量可以是受限的
public class C1
public class C2
第一个T变量必须继承Number,第二个T必须继承Person和实现Comparable
3.Generics方法
像Generics类一样,方法和构造函数也可以有类型参数。方法的参数的返回值都可以有类型参数,进行Generics。
//Example 4
1 public
2 if (t1.compareTo(t2) > 0)
3 return t1;
4 else return t2;
5 }
这里,max方法的参数类型为单一的T类型,而T类型继承了Comparable,max的参数和返回值都有相同的超类。下面的Example 5显示了max方法的几个约束。
//Examp