JDK5以前,对象保存到集合中就会失去其特性,取出时通常要程序员手工进行类型的强制转换,这样不可避免就会引发程序的一些安全性问题。
泛形的典型应用:BaseDao
例如:
public class Demo1 { /** * @param args */ @Test public void test(){ ArrayList list = new ArrayList(); list.add("abc"); list.add(new Integer(10)); //list.add("name"); list.add(new Random()); list.add(new ArrayList()); for(int i=0;i<list.size();i++){ Integer num = (Integer) list.get(i); //运行时会出错,但编码时发现不了 System.out.println(num); } }
使用泛型,对list集合中存放的对象加了限制,此时再放入其他的对象就出现了错误
@Test public void testt(){ ArrayList<Integer> list = new ArrayList<Integer>(); list.add(new Integer(10)); /*在list中放入非Integer类型讲提示错误 * list.add("name"); list.add(new Random()); list.add(new ArrayList());*/ for(int i=0;i<list.size();i++){ Integer num = list.get(i); //不用强转 System.out.println(num); } }
泛型的作用
泛形的基本术语
以ArrayList<E>为例:
泛型典型应用
public void test1(){ List<String> list=new ArrayList<String>(); list.add("abc"); list.add("1"); list.add("efg"); //1. Iterator<String> it=list.iterator(); while(it.hasNext()){ String str=it.next(); System.out.println(str); } }
@Test public void test1(){ List<String> list=new ArrayList<String>(); list.add("abc"); list.add("1"); list.add("efg"); //2. for(String str:list){ System.out.println(str); } }
@Test //未加泛型 public void test2(){ Map m=new HashMap(); m.put(1,"aa"); m.put(2,"bb"); m.put(3,"cc"); //map.keySet(); map.entrySet(); Set set=m.entrySet(); Iterator it=set.iterator(); while(it.hasNext()){ Map.Entry me=(Map.Entry)it.next(); System.out.println(me.getKey()+"..."+me.getValue()); } } @Test //加泛型 public void test3(){ Map<Integer,String> map=new HashMap<Integer,String>(); map.put(1,"aa"); map.put(2,"bb"); map.put(3,"cc"); Set<Map.Entry<Integer,String>> set= map.entrySet(); Iterator<Map.Entry<Integer,String>> it=set.iterator(); while(it.hasNext()){ Map.Entry<Integer,String> me=it.next(); System.out.println(me.getKey()+"..."+me.getValue()); } }
使用泛形时的几个常见问题:
public static void test4(ArrayList list){ } public static void test5(ArrayList<String> list){ } public static void main(String[] args) { test5(new ArrayList()); }
自定义泛形——泛型方法
Java程序中的普通方法、构造方法和静态方法中都可以使用泛型。
方法使用泛形前,必须对泛形进行声明,
语法:
<T> ,T可以是任意字母,但通常必须要大写。
package com.hbsi.generics; import java.util.Arrays; public class Demo3 { /** * @param args */ public static void main(String[] args) { Integer arr1[]={1,2,3,4}; test(arr1,0,3); /*for循环遍历 * for(Integer i:arr1){ System.out.print(i); }//4231*/ //测试Test System.out.println(Arrays.asList(arr1)); //测试Text2 String arr[]={"a","b","c","d"}; test2(arr); System.out.println(Arrays.asList(arr)); } //编写一个泛形方法,实现数组元素的交换。 public static <T> void test(T arr[],int i,int j){ T temp=arr[i]; arr[i]=arr[j]; arr[j]=temp; } //编写一个泛形方法,接收一个任意数组,并颠倒数组中的所有元素 public static <T> void test2(T arr[]){ int start=0; int end=arr.length-1; while(true){ if(start>=end){ break; } T temp=arr[start]; arr[start]=arr[end]; arr[end]=temp; start++; end--; } } }
package com.hbsi.generics; public class Demo2 <T,E>{ /** * @param args */ public static void main(String[] args) { // TODO Auto-generated method stub } //自定义泛型方法 public void test1(T t){ } public void test2(T t){ } public static <T> void test3(T t){ } }
通配符
定义一个方法,接收一个集合,并打印出集合中的所有元素,如下所示:
void print (Collection<String> c) { for (String e : c) { System.out.println(e); } }
问题:该方法只能打印保存了String对象的集合,不能打印其它集合。通配符用于解决此类问题,
方法的定义可改写为如下形式:
void print (Collection<?> c) { //Collection<?>(发音为:"collection of unknown") for (Object e : c) { System.out.println(e); } }
此种形式下需要注意的是:
由于print方法c参数的类型为Collection<?>,即表示一种不确定的类型
因此在方法体内不能调用与类型相关的方法,例如add()方法。但可以调用与类型无关的方法,例如size()方法
总结:使用?通配符主要用于引用对象,使用了?通配符,就只能调对象与类型无关的方法,不能调用对象与类型有关的方法。
有限制的通配符
限定通配符的上边界(?的类型必须是Number的子类),限定通配符总是包括自己。
正确:Vector<? extends Number> x = new Vector<Integer>();
错误:Vector<? extends Number> x = new Vector<String>();
限定通配符的下边界(?的类型必须是Integer的父类)
正确:Vector<? super Integer> x = new Vector<Number>();
错误:Vector<? super Integer> x = new Vector<Byte>();