泛型
它是在JavaSE5中引入的特性。泛型实现了类型参数化的概念,使代码可以应用于多种类型。使用泛型之场景之一就是创建容器类。如下:
// 使用类型参数(即紧跟类名后尖括号中的类名)创建容器且容器只能存放String对象
List<String> stringList = new ArrayList<String>();
stringList.add("str1");
stringList.add(1); // 编译错误:List<String>不能存放int
String str = stringList.get(0); // 不需要进行类型转换
// 不使类型参数时如下
List list = new ArrayList();
list.add("str1");
String item = (String)list.get(0); // 进行类型转换
使用泛型创建容器类时的好处,第一能在编译期检查容器中的类型是否正确;第二在调用容器的方法时不需要转换成指定的参数类型。
泛型接口和类定义
在接口名称或类名称后添加尖括号及类型参数名称,多个类型参数用逗号分隔
// 泛型类
class Class<T> {
}
// 泛型接口
interface Interface<G> {
}
泛型方法
这个方法所在的类可以是泛型类,也可以不是泛型类。无论何时,只要你能做到,你就应该尽量使用泛型方法,因为它可以使事情更清楚明白。定义泛型方法,只需要将泛型参数列表置于返回值之前,如下:
import java.util.ArrayList;
import java.util.Date;
public class GeneratorMethods {
public <T, V, U> void f(T x, V y, U u) {
System.out.println(x.getClass().getName() + "=="
+ y.getClass().getName() + "==" + u.getClass().getName());
}
public static void main(String[] args) {
GeneratorMethods gm = new GeneratorMethods();
gm.f("", new ArrayList<Integer>(), new Long(10));
gm.f(1, new Date(), "str");
}
}
当使用泛型类时,必须在创建对象的时候指定类型参数的值,而使用泛型方法的时候,通常不必指明参数类型,因为编译器会为我们找出具体的类型。
类型参数的擦除
在泛型代码内部,无法获得任何有关泛型参数类型的信息。下面程序进行泛型类的参数类型的输出。
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
class Frob {}
class Fnorkle {}
class Quark<Q> {}
class Particle<POSITION, MOMENTUM> {}
public class LostInformation {
public static void main(String[] args) {
List<Frob> list = new ArrayList<Frob>();
// Arrays.toString():输出数组的内容
// Class.getTypeParameters():输出泛型类的参数类型
System.out.println(Arrays.toString(list.getClass().getTypeParameters()));
Map<Frob, Fnorkle> map = new HashMap<Frob, Fnorkle>();
System.out.println(Arrays.toString(map.getClass().getTypeParameters()));
Quark<Fnorkle> quark = new Quark<Fnorkle>();
System.out.println(Arrays.toString(quark.getClass().getTypeParameters()));
Particle<Long, Double> p = new Particle<Long, Double>();
System.out.println(Arrays.toString(p.getClass().getTypeParameters()));
}
}
输出结果为:
[E]
[K, V]
[Q]
[POSITION, MOMENTUM]
在这些输出的结果中,能够发现的只是用作参数占位符的标识符,而不是实例化泛型类时转入类的类型参数,这并非有用的信息。
边界
边界使得你可以在用于泛型的参数类型上设置限制条件。使用无界泛型参数时,只能调用Object类中方法,而有边界的类型参数可以调用一个指定的子集。
如下所示
import java.awt.Color;
interface HasColor {
Color getColor();
}
// 泛型重用了extends关键字,这里表示“T”为HasColor的任何子类
class Colored<T extends HasColor> {
T item;
Colored(T item) {
this.item = item;
}
// 因为确定了“T”是HasColor的子类,所以可用调用HasColor中的方法
Color color() {
return item.getColor();
}
}