一、前言
泛型这个词在现在的JAVA挺时髦,光从字面上你是无法知道它代表些什么东东的,所以我们还是不要从字面去理解,而是从一些实例去了解它吧。
二、泛型之前的日子
JDK1.4之前是没有泛型的概念的,所以我们才会有下面的代码:
List list = new ArrayListfl; 1ist. add(”aaaa”); list. add(rrbbbbnr)s; list. add(”cccc”); for (Iterator it = list.iterator'; it.hasNextfl;) String str = (String) it.nextfl; System. out. println(str);
上面是一段很平常的代码,在一个List集合加入一些字符串,然后再用一个遍历循环把它打印出来。“String str = (String) it.next()”这一句我们可以看到List取出值都是Object,所以我们要得String型,还要做一个类型转换,真是麻烦。更麻烦的是list.add(Object obj)的参数是Object类型,所以如果我们一不小心把list.add("cccc");写成list.add(new Integer(76));程序在循环打印的类型转换中就会出错。
问题:我们能不能让add方法只认String型呢?
回答:可以!用JDK5.0的泛型。
三、泛型后的幸福生活
JAVA有了泛型后,就象十年的老光棍讨了老婆,那个好处自不待言。我们来看看上面的例子改成泛型的写法是怎么样的:
Listcstring> list = new Arraytist<string>O; list.add(”saaa"); liat.add(”bbbb”)l;; list.add(”cccc’9l;; for (Iterator<string> it = list.iteratorfl; it.hasNextOs;) string strsit.next C); Systenn.out.println(str);
看到差别了吗?泛型其实很简单,就是在定义类型的后面加上"<类型>"这样子的声明就行了,它主要还有以下差别:
• list.add方法只能接受String类型。list.add(new Integer(76))这样的语句不需要运行程序,在编译时就会检查通不过。
• it.next()的返回值不再是Object,而变成了String
当然我们其实在循环部份也可以象下面这么写,是不是简洁了很多呢 :-)
List<String> list = new ArrayList<String>(); list.add("aaaa"); list.add("bbbb"); list.add("cccc"); for (String str : list) { System.out.println(str); }
当然需要说明的是,List不仅可以List<String>,也可以是List<Integer>等等其他任何类型。
四、更深入了解泛型
(1)层层推进的泛型声明
“List<List> list;”表示什么呢?就是只接收List型的参数,比如:
List<List> list = new ArrayList<List>(); list.add(new ArrayList()); list.add(new Vector()); list.add(new LinkedList());
这里要注意List是接口,ArrayList、Vector、LinkedList都是这一接口下的实现类。下面这个有点怪异了,“List<List<String>> list;”表示它只接受List型的参数,而且这种List型的参数又是只是只接受String型,有点层层推进的味道在里面了。
ListCistcstring>> list = new ArrayListCist<string>>O; list.add(new ArraytistcString>Q); list.add(new vector<String>() ; list.add(new LinkedList<string>());
(2)使用泛型上限通通配符:extends
这里要着重强调一点:变量的泛型声明和方法的参数的泛型声明有很大差别。
变量声明成某类型,同时也可以接受它的子类。比如说Integer、Long、Float都是抽象类Number的子类,所以下面的代码一点问题也没有:
List<Nuzthers 1ist = net.i ArrayList<Nunber>fl; list.add(new Integer(1)); list.add(new Long(1)); list.add(neN Float(1.2)!);t
但如果换成方法参数的泛型声明则要严格得多了:子类也是不行的。比如下面的代码就是错误的,因为printList参数只接受Number值的List,就是是Number子类的Integer值的List也不行。
public static void main(String[] args) List<Integer> list = new ArrayList<Integer>Q; list.add(new Integer(1)); list.add(new Integer(2)); printIist (list); private static void printlist(List<Nunber> list) { for (Number nuni list) System. out . println (nun’i) ;
上面代码修改的方法有两个,如下
修改方法一:改变量的泛型声明
将 List<Integer> list = new ArrayList<Integer>();
改为 List<Number> list = new ArrayList<Number>();
修改方法二:用界限通配符改方法参数的泛型声明
将 printList(List<Number> list)
改为 printList(List<? extends Number> list)
说明:extends 的含义就是表示参数可以接受Number型的子类。
(3)使用泛型下限通通配符:super
在上限就有下限,下限行就是super,用法和extends一样,含义则和extends相反。比如printList(List<? super Integer> list)表示参数可以接受Integer型及Integer型的超类,即Number了,当然也包括Object这个顶级类。
(4)配置符:?
?表示可以接受任何类型,不过我觉得它用得不多,因为printList(List<?> list)和printList(List list)的作用是一样的。
五、创建一个支持泛型的类
(1)创建一个泛型的类
public cla5s Point<T> T x; T y; public T getx( { return x; public T gety( return y: publie void 5etX(T x) thia.x = x; public void setY(T 7) { this.y =
使用这个类的代码如下:
Point<Integer> p = new Point<Integer>Q; p.setX(new Integer(1)): p.setY(new Integer (2))n;r Pointxstring> b = new PointrString>lf;? b.setX(”1”) b.setY(”2”) ;
说明:在Point<T>的定义中,T并非关键字,你也可以这样定义Point<ABC>,当然一般还是写T吧,简单也规范。
(2)泛型类的继承与实现
java.util.Comparator类是JDK里用来排序的,其源代码如下:
package java. util; public interface comparator<T> ( mnt connpare(T o1, T o2h; boolean equals(Object obj);
一个实现此接口的类如下:
public class MyComparator<T> implements Comparator<ObjectInstance> { public int compare(ObjectInstance o1, ObjectInstance o2) { String s1 = o1.getObjectName().getCanonicalName(); String s2 = o2.getObjectName().getCanonicalName(); return s1.compareToIgnoreCase(s2); } }
说明:ObjectInstance可能大家还太明白,这是我实际项目中的一段代码(关于JMX的),ObjectInstance全称javax.management.ObjectInstance。MyComparator的使用代码如下:
Setset= . ListczObjectinstance> mbeans = new ArrayList<Objectlnstance>(set); Collections.sort(mbeans, new MyComparator<Objectlnstance>O);
六、最后的感言
JAVA有了泛型就象老光棍讨了老婆,好处大大的,但和女人一样麻烦也跟着来了:它的严格类型检查,使隐藏的BUG更少。有些地方确实也使代码简洁了,有些地方却会使得代码更复杂。所以运用之妙在于是否用得适当,尽量把泛型往简单里用,别越搞越复杂了。