java1.5新特性 (二)

2014-11-24 09:04:21 · 作者: · 浏览: 1
ctionsExample() {
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 = new ArrayList();
list.add(new String("test string"));
// list.add(new Integer(9)); this no longer compiles
inspectCollection(list);
}
protected void inspectCollection(Collection aCollection) {
Iterator i = aCollection.iterator();
while(i.hasNext()) {
String element = i.next();
}
}

protected void collectionsExample() {
ArrayList list = new ArrayList();
list.add(new String("test string"));
// list.add(new Integer(9)); this no longer compiles
inspectCollection(list);
}
protected void inspectCollection(Collection aCollection) {
Iterator i = aCollection.iterator();
while(i.hasNext()) {
String element = i.next();
}
}


从上面第2行我们在创建ArrayList时使用了新语法,在JDK1.5中所有的Collection都加入了Generics的声明。例:


[java]
public class ArrayList extends AbstractList {
// details omitted...
public void add(E element) {
// details omitted
}
public Iterator iterator() {
// details omitted
}
}

public class ArrayList extends AbstractList {
// details omitted...
public void add(E element) {
// details omitted
}
public Iterator 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 T max(T t1, T t2) {
2 if (t1.compareTo(t2) > 0)
3 return t1;
4 else return t2;
5 }

  这里,max方法的参数类型为单一的T类型,而T类型继承了Comparable,max的参数和返回值都有相同的超类。下面的Example 5显示了max方法的几个约束。

//Examp