Java 基础篇专栏 – 泛型
背景
在没有泛型前,一旦把一个对象丢进集合中,集合就会忘记对象的类型,把所有的对象都当成 Object 类型处理。当程序从集合中取出对象后,就需要进行强制类型转换,这种转换很容易引起 ClassCastException 异常。
定义
程序在创建集合时指定集合元素的类型。增加了泛型支持后的集合,可以记住集合中元素的类型,并可以在编译时检查集合中元素的类型,如果试图向集合中添加不满足类型要求的对象,编译器就会报错。
示例
集合使用泛型
1import java.util.ArrayList;
2import java.util.HashMap;
3import java.util.List;
4import java.util.Map;
5
6public class DiamondTest {
7 public static void main(String[] args) {
8 List<String> books = new ArrayList<>();
9 books.add("learn");
10 books.add("java");
11 books.forEach(book -> System.out.println(book.length()));
12
13 Map<String, List<String>> schoolsInfo = new HashMap<>();
14 List<String> schools = new ArrayList<>();
15 schools.add("i");
16 schools.add("love");
17 schoolsInfo.put("java", schools);
18 schoolsInfo.forEach((key, value) -> System.out.println(key + "--->" + value));
19 }
20}
类、接口使用泛型
1public class Apple<T> {
2 private T info;
3 public Apple() {}
4 public Apple(T info) {
5 this.info = info;
6 }
7 public void setInfo(T info) {
8 this.info = info;
9 }
10 public T getinfo() {
11 return this.info;
12 }
13 public static void main(String[] args) {
14 Apple<String> a1 = new Apple<>("Apple");
15 System.out.println(a1.getinfo());
16 Apple<Double> a2 = new Apple<>(5.67);
17 System.out.println(a2.getinfo());
18 }
19}
类型通配符
需求分析
1public void test(List<Object> c) {
2 for (int i = 0; i < c.size(); i++) {
3 System.out.prinln(c.get(i));
4 }
5}
这个方法声明没有任何问题,但是调用该方法实际传入的参数值,可能会出错。考虑如下代码:
1List<String> strList = new ArrayList<>();
2test(strList); // 编译出错,表明 List<String> 对象不能被当成 List<Object> 对象使用,也就是说 List<String> 并不是 List<Object> 的子类。
问题解决
为了表示各种泛型 List 的父类,可以使用类型通配符。List<?>
表示元素类型未知的 List。这个 ?
号被称为通配符,它可以匹配任何类型。将上面的代码,改为如下形式:
1public void test(List<?> c) {
2 for (int i = 0; i < c.size(); i++) {
3 System.out.prinln(c.get(i));
4 }
5}
现在传入任何类型的 List,程序可以正常打印集合 c 中的元素。
不过集合中元素的类型会被当成 Object 类型对待。
类型通配符的上限
需求分析
当使用 List<?>
时,表明它可以是任何泛型 List 的父类。如果我们只希望它代表某一类泛型 List 的父类,java 提供了被限制的泛型通配符。
看如下代码:
1public abstract class Shape {
2 public abstract void draw(Canvas c);
3}
1public class Circle extends Shape {
2 @Override
3 public void draw(Canvas c) {
4 System.out.println("在画布上" + c + "上画一个圆");
5 }
6}
1public class Rectangle extends Shape {
2 @Override
3 public void draw(Canvas c) {
4 System.out.println("把一个矩形画在画布" + c + "上");
5 }
6}
1import java.util.List;
2
3public class Canvas {
4 public void drawAll(List<Shape> shapes) {
5 for (Shape s : shapes) {
6 s.draw(this);
7 }
8 }
9}
下面代码将引起编译错误,因为 List<Circle>
并不是 List<Shape>
的子类型,所以不能把 List<Circle>
对象当成 List<Shape>
类用。
1// 错误示范
2List<Circle> circleList = new ArrayList<>();
3Canvas c = new Canvas();
4c.drawAll(circleList);
问题解决
方法一:通过类型通配符解决,即 List<?>
方式。
需要进行强制类型转换,因为
List<?>
中元素默认为 Object 类型
1import java.util.List;
2
3public class Canvas {
4 public void drawAll(List<?> shapes) {
5 for (Object obj : shapes) {
6 Shape s = (Shape)obj // 但是需要进行强制类型转换,因为前面提到过 List<?> 中元素默认为 Object 类型
7 s.draw(this);
8 }
9 }
10}
方法二:使用被限制的泛型通配符
List<? extends Shape>
可以表示List<Circle>
和List<Rectangle>
的父类。只要 List 尖括号里的类型是 Shape 的子类即可。
1import java.util.List;
2
3public class Canvas {
4 public void drawAll(List<? extends Shape> shapes) { // 使用被限制的泛型通配符
5 for (Shape s : shapes) {
6 s.draw(this);
7 }
8 }
9}
形参类型上的应用
在定义类型形参时设定类型通配符上限。以此来表示传递给该类型形参的实际类型必须是该上限类型或者其子类。
1public class Apple<T extends Number> {
2 T col;
3 public static void main(String[] args) {
4 Apple<Integer> ai = new Apple<>();
5 Apple<Double> ad = new Apple<>();
6 Apple<String> as = new Apple<>(); // 编译出错,试图将 String 类型传给 T 形参,但是 String 不是 Number 的子类型
7 }
8}
泛型方法
定义
泛型方法就是在声明方法时定义一个或多个类型形参。多个类型参数之间用逗号隔开。
1修饰符 <T, S> 返回值类型 方法名(形参列表) {方法体}
需求分析
泛型方法解决了什么问题?
1static void fromArrayToCollection(Object[] a, Collection<Object> c) {
2 for (Object o : a) {
3 c.add(o)
4 }
5}
上面定义的方法没有任何问题,关键在于方法中的 c 形参,它的数据类型是 Collection<Object>
。假设传入的实际参数的类型是 Collection<String>
,因为 Collection<String>
并不是 Collection<Object>
的子类,所以这个方法的功能非常有限,它只能将 Object[] 数组的元素复制到元素为 Object 类型(Object 的子类不行)的 Collection 集合中。
如果使用通配符 Collection<?>
是否可行呢?显然也不行,Collection 集合提供的的 add() 方法中有类型参数 E,而如果使用类型通配符,这样程序无法确定 c 集合中的元素类型,所以无法正确调用 add 方法。
问题解决
泛型方法。
1import java.util.ArrayList;
2import java.util.Collection;
3
4public class GenericMethodTest {
5 // 声明一个泛型方法
6 static <T> void fromArryToCollection(T[] a, Collection<T> c) {
7 for (T o : a) {
8 c.add(o);
9 }
10 }
11 public static void main(String[] args) {
12 Object [] oa = new Object[100];
13 Collection<Object> co = new ArrayList<>();
14 fromArryToCollection(oa, co);
15
16 String[] sa = new String[100];
17 Collection<String> cs = new ArrayList<>();
18 fromArryToCollection(sa, cs);
19 }
20}
进一步改造,如下
1import java.util.ArrayList;
2import java.util.Collection;
3import java.util.List;
4
5public class RightTest {
6 static <T> void test(Collection<? extends T> from, Collection<T> to) {
7 for (T ele : from) {
8 to.add(ele);
9 }
10 }
11 public static void main(String[] args) {
12 List<String> as = new ArrayList<>();
13 List<Object> ao = new ArrayList<>();
14 test(as, ao);
15 }
16}
泛型构造器
和泛型方法类似,Java 也允许在构造器签名中声明类型形参,这就产生了所谓的泛型构造器
1public class Foo {
2 // 泛型构造器
3 public <T> Foo(T t) {
4 System.out.println(t);
5 }
6}
7
8public class GenericConstructor {
9 public static void main(String[] args) {
10 new <String> Foo("crazy");
11 new Foo("crazy"); // 与上面等价
12 new <Sting> Foo(12.3) // 出错
13 }
14}
类型通配符下限
需求分析
实现一个方法,将 src 集合里的元素复制到 dest 集合里,且返回最后一个被复制的元素的功能。
因为 dest 集合可以保存 src 集合里的所有元素,所以 dest 集合元素的类型应该是 src 集合元素类型的父类。
为了表示两个参数间的类型依赖,考虑同时使用之前介绍过的通配符、泛型参数来实现该方法,代码如下:
1public static <T> T copy(Collection<T> dest, Collection<? extends T> src) {
2 T last = null;
3 for (T ele : src) {
4 last = ele;
5 dest.add(ele);
6 }
7 return last;
8}
9
10List<Number> ln = new ArrayList<>();
11List<Integer> li = new ArrayList<>();
12// 下面代码会引起编译错误
13Integer last = copy(ln, li);
上面的代码有一个问题,ln 的类型是 List<Number>
,那么 T 的实际类型就是 Number,即返回值 last 类型是 Number 类型。但实际上最后一个复制元素的类型一定是 Integer。也就是说,程序在复制集合元素的过程中,丢失了 src 集合元素的类型。
问题解决
为了解决这个问题,引入通配符下限,<? super Type>
。表示必须是 Type 本身或者 Type 的父类。改写后的完整代码,如下:
1import java.util.ArrayList;
2import java.util.Collection;
3import java.util.List;
4
5public class MyUtils {
6 // 使用通配符下限
7 public static <T> T copy(Collection<? super T> dest, Collection<T> src) {
8 T last = null;
9 for (T ele : src) {
10 last = ele;
11 dest.add(ele);
12 }
13 return last;
14 }
15 public static void main(String[] args) {
16 List<Number> ln = new ArrayList<>();
17 List<Integer> li = new ArrayList<>();
18 li.add(5);
19 // 此时可以准确知道最后一个被复制的元素是 Integer 类型,而不是笼统的 Number 类型
20 Integer last = copy(ln, li);
21 System.out.println(last);
22 System.out.println(ln);
23 }
24}
泛型擦除
泛型基本上都是在编译器这个层次来实现的。在生成的 Java 字节代码中是不包含泛型中的类型信息的。使用泛型的时候加上的类型参数, 会被编译器在编译的时候去掉。这个过程就称为泛型擦除。如在代码中定义的 List<Object>
和 List<String>
等类型, 在编译之后都会变成 List, JVM 看到的只是 List, 泛型附加的类型信息对 JVM 来说是不可见的。
原文始发于微信公众号(SRE工程师):Java 基础篇专栏 - 泛型