泛型是J2SE 5.0最重要的特性。他们让你写一个type(类或接口)和创建一个实例通过传递一个或多个引用类型。这个实例受限于只能作用于这些类型。比如,在java 5,java.util.List 已经被泛化。当建立一个list对象时,你通过传递一个java类型建立一个List实例,此list实例只能作用于所传递的类型。这意味着如果你传递一个String ,此List实例只能拥有String对象;如果你传递一个Integer,此实例只能存贮Integer对象。除了创建参数化的类型,你还能创建参数化的函数。 没有泛型的日子 举个简单的例子,在JDK 5.0的之前版本中,类List的函数add接受一个Object参数:
public java.lang.Object get(int index) throws IndexOutOfBoundsException
List stringList1 = new ArrayList(); stringList1.add("Java 5"); stringList1.add("with generics"); 当你想从stringList1取得一个元素时,你得到了一个Object.为了操作原来的类型元素,你不得不把它转换为String。 String s1 = (String) stringList1.get(0); 但是,假如你曾经把一个非String对象加入stringList1中,上面的代码会抛出一个ClassCastException. 有了泛型,你能创建一个单一用途的List实例.比如,你能创建一个只接受String对象的List实例,另外一个实例只能接受Employee对象.这同样适用于集合框架中的其他类型. 泛型入门 像一个函数能接受参数一样,一个泛型也能接受参数.这就是一个泛型经常被称为一个参数化类型的原因.但是不像函数用()传递参数,泛型是用<>传递参数的.声明一个泛型和声明一个普通类没有什么区别,只不过你把泛型的变量放在<>中.
boolean add<E o>E get(int index)
-----------------------------译者添加-------------------- 如果你传递一个String给一个List,比如:
List<String> myList;
NOTE:根据惯例,我们使用一个唯一的大写字目表示一个类型变量。为了创建一个泛型,你需在声明时传递同样的参数列表。比如,你要想创建一个ArrayList来操作String ,你必须把String放在<>中。如:
List<String> myList = new ArrayList<String>();
public interface Map<K,V>
V put(K key, V value)Collection<V> values()
列表1的例子将比较List在JDK 1.4 和JDK1.5的不同
package com.brainysoftware.jdk5.app16; import java.util.List; import java.util.ArrayList; public class GenericListTest { public static void main(String[] args) { // in JDK 1.4 List stringList1 = new ArrayList(); stringList1.add("Java 1.0 - 5.0"); stringList1.add("without generics"); // cast to java.lang.String String s1 = (String) stringList1.get(0); System.out.println(s1.toUpperCase()); // now with generics in JDK 5 List<String> stringList2 = new ArrayList<String>(); stringList2.add("Java 5.0"); stringList2.add("with generics"); // no need for type casting String s2 = stringList2.get(0); System.out.println(s2.toUpperCase()); }}
NOTE:泛型的类型检查(type checking)是在编译时完成的. 最让人感兴趣的事情是,一个泛型是个类型并且能被当作一个类型变量。比如,你想你的List储存lists of Strings,你能通过把List<String>作为他的类型变量来声明List。比如:
List<List<String>> myListOfListsOfStrings;
String s = myListOfListsOfStrings.get(0).get(0);
package com.brainysoftware.jdk5.app16; import java.util.ArrayList; import java.util.List; public class ListOfListsTest { public static void main(String[] args) { List<String> listOfStrings = new ArrayList<String>(); listOfStrings.add("Hello again"); List<List<String>> listOfLists = new ArrayList<List<String>>(); listOfLists.add(listOfStrings); String s = listOfLists.get(0).get(0); System.out.println(s); // prints "Hello again" }}
package com.brainysoftware.jdk5.app16; import java.util.HashMap; import java.util.Map; public class MapTest { public static void main(String[] args) { Map<String, String> map = new HashMap<String, String>(); map.put("key1", "value1"); map.put("key2", "value2"); String value1 = map.get("key1"); }}
没有参数的情况下使用泛型 既然在J2SE 5.0中收集类型已经泛型化,那么,原来的使用这些类型的代码将如何呢?很幸运,他们在JAVA 5中将继续工作,因为你能使用没有参数的泛型。比如,你能继续像原来一样使用List接口,正如下面的例子一样。
List stringList1 = new ArrayList(); stringList1.add("Java 1.0 - 5.0"); stringList1.add("without generics"); String s1 = (String) stringList1.get(0);
尽管如此,一个需要注意的事情是,JDK5编译器希望你使用带参数的泛型。否则,编译器将提示警告,因为他认为你可能忘了定义类型变量s。比如,编译上面的代码的时候你会看到下面这些警告,因为第一个List被认为是原型。 Note: com/brainysoftware/jdk5/app16/GenericListTest.java 当你使用原型时,如果你不想看到这些警告,你有几个选择来达到目的: 使用 ? 通配符
List<Object> list1 = new ArrayList<Object>(); List<String> list2 = new ArrayList<String>();
import java.util.ArrayList;import java.util.List; public class AllowedTypeTest { public static void doIt(List<Object> l) { } public static void main(String[] args) { List<String> myList = new ArrayList<String>(); // 这里将产生一个错误 doIt(myList); }} 上面的代码无法编译,因为你试图传递一个错误的类型给函数doIt。doIt的参数是List<Object>二你传递的参数是List<String>。
public static void doIt(List<?> l) {}
import java.util.ArrayList;import java.util.List; public class WildCardTest { public static void printList(List<?> list) { for (Object element : list) { System.out.println(element); } } public static void main(String[] args) { List<String> list1 = new ArrayList<String>(); list1.add("Hello"); list1.add("World"); printList(list1); List<Integer> list2 = new ArrayList<Integer>(); list2.add(100); list2.add(200); printList(list2); }}
List<?> myList = new ArrayList<?>(); // 不合法
List<Object> myList = new ArrayList<Object>();
你能使用原型或使用通配符,但这样无法在编译时进行安全类型检查,因为你能传递任何类型的List,比如List<String>的实例。你可以使用List<Number>作为参数,但是你就只能传递List<Number>给函数。但这样就使你的函数功能减少,因为你可能更多的时候要操作List<Integer>或List<Long>,而不是List<Number>。 J2SE5.0增加了一个规则来解决了这种约束,这个规则就是允许你定义一个上界(upper bound) 类型变量.在这种方式中,你能传递一个类型或它的子类。在上面getAverage函数的例子中,你能传递一个List<Number>或它的子类的实例,比如List<Integer> or List<Float>。 使用上界规则的语法这么定义的:GenericType<? extends upperBoundType>. 比如,对getAverage函数的参数,你可以这么写List<? extends Number>. 下面例子说明了如何使用这种规则。
import java.util.ArrayList;import java.util.List; public class BoundedWildcardTest { public static double getAverage(List<? extends Number> numberList) { double total = 0.0; for (Number number : numberList) total += number.doubleValue(); return total/numberList.size(); } public static void main(String[] args) { List<Integer> integerList = new ArrayList<Integer>(); integerList.add(3); integerList.add(30); integerList.add(300); System.out.println(getAverage(integerList)); // 111.0 List<Double> doubleList = new ArrayList<Double>(); doubleList.add(3.0); doubleList.add(33.0); System.out.println(getAverage(doubleList)); // 18.0 }} 由于有了上界规则,上面例子中的getAverage函数允许你传递一个List<Number> 或一个类型变量是任何java.lang.Number子类的List。 下界规则 创建泛型 前面的章节主要说明了如何使使用泛型,特别是集合框架中的类。现在我们开始学习如何写自己的泛型。 基本上,除了声明一些你想要使用的类型变量s外,一个泛型和别的类没有什么区别。这些类型变量s位于类型后面的<>中。比如,下面的Point就是个泛型。一个Point对象代表了一个系统中的点,它有横坐标和纵坐标。通过使Point泛型化,你能定义一个点实例的精确程度。比如,如果一个Point对象需要非常精确,你就把Double作为类型变量。否则,Integer 就够了。
public class Point<T> { T x; T y; public Point(T x, T y) { this.x = x; this.y = y; } public T getX() { return x; } public T getY() { return y; } public void setX(T x) { this.x = x; } public void setY(T y) { this.y = y; }}
Point<Integer> point1 = new Point<Integer>(4, 2); point1.setX(7); Point<Double> point2 = new Point<Double>(1.3, 2.6); point2.setX(109.91);
你已经知道通过传递不同类型的类型变量给泛型可以产生不同的JAVA类型。就是说List<String>和List<Object>的类型是不同的。尽管String是java.lang.Object。但是传递一个List<String>给一个参数是List<Object>的函数会参数会产生编译错误(compile error)。函数能用 ? 通配符使其接受任何类型的参数。List<?> 意味着任何类型的对象。 |
JDK5.0泛型
最新推荐文章于 2020-08-15 17:26:04 发布
JDK5.0泛型(尖括号?看这里)