集合没有泛型的时候,集合存放数据时都会丢失原来的类型,全部改为Object。这样可以获得良好的通用性。但是取出的时候,就需要做类型转换,如果类型写错了,转换就会出现异常。为了有更好的安全性和可读性,Java在JDK1.5的时候加入了泛型。
1.在集合(List、Set、Map)上使用泛型
在通用类或者接口上使用泛型
在方法上使用泛型
明白什么是泛型擦除
泛型的作用
使用泛型机制编写的程序代码要比那些杂乱地使用Object 变量 ,然后再进行强制类型转换的代码具有更好的安全性和可读性 。泛型对于集合类尤其有用 ,例如 ,ArrayList就是一个无处不在的集合类
没有泛型的代码:
list.add(123);
list.add("abc");
list.add(1>2);
list.add('E');
list.add(890.12);
for (int i = 0; i < list.size(); i++) {
System.out.println( list.get(i) );
}
输入什么类型,就输出什么类型。但是我希望list里面存放的数据类型只有字符串怎么办 ?没有代码的情况下代码是这样的:
List list = new ArrayList();
list.add("123.A");
list.add("abc.def");
for (int i = 0; i < list.size(); i++) {
String str = (String) list.get(i);
System.out.println( str.split("\\.")[0] ) ;
}```
但是list中如果有其他类型呢?
List list = new ArrayList();
list.add("123.A");
list.add(123.123); // 这行数据就是一个浮点型
list.add("abc.def");
for (int i = 0; i < list.size(); i++) {
String str = (String) list.get(i);
System.out.println( str.split("\\.")[0] ) ;
}
为了解决这类问题,使用泛型是不二之选。通过泛型可以限制集合中数据的类型,只有符合的类型才能放到集合中
格式
在声明的集合类型后面跟上一对尖括号,实现的类型构造器的小括号前面跟上#####一对尖括号。里面写上需要存放的数据类型
可以看到,在定义泛型后,添加 123.123 浮点数时编译器就开始报错了。在使用泛型后,代码也不需要做类型强转了。
List<String> list = new ArrayList<String>();
list.add("123.A");
list.add("abc.def");
for (int i = 0; i < list.size(); i++) {
String str = list.get(i);
System.out.println( str.split("\\.")[0] ) ;
}
Map演示
Map<String, String> data = new HashMap<>();
data.put("name", "张三");
data.put("age", "11岁");
data.put("sex", "男");
for (Map.Entry<String, String> entry: data.entrySet()) {
System.out.println( entry.getKey() +":"+ entry.getValue());
}
简单的泛型类
在定义类Pair时在类名后跟上<T>
public class Pair<T> {
private T first;
private T second;
public Pair() {
first = null;
second = null;
}
public Pair(T first , T second){
this.first = first;
this.second = second;
}
public void setFirst(T newValue) {
first = newValue;
}
public void setSecond(T newValue) {
second = newValue;
}
public T getFirst() {
return first;
}
public T getSecond() {
return second;
}
@Override
public String toString() {
return "Pair [first=" + first + ", second=" + second + "]";
}
}
使用
public static void main(String[] args) {
System.out.println(new Pair<Integer>(1, 2));
//输出: Pair [first=1, second=2]
System.out.println(new Pair<String>("诸葛", "孔明"));
//输出: Pair [first=诸葛, second=孔明]
}
使用泛型,就如同将原来类定义的 T 替换为了指定的类型版本一样,
比如:
private Integer first;
private Integer second;
public Pair() {
first = null;
second = null;
}
public Pair(Integer first , Integer second){
this.first = first;
this.second = second;
}
//....
}
类型参数就跟在方法或构造函数中普通的参数一样。就像一个方法有形式参数 (formal value parameters)来描述它操作的参数的种类一样,一个泛型声明也有形式类型参数(formal type parameters)。当一个方法被调用,实参(actual arguments)替换形参,方法体被执行。当一个泛型声明被调用,实际类型参数(actual type arguments)取代形式类型参数。
泛型方法
在使用前,我们先明确一下泛型的各种通配符:
1. T:type 数据类型
2. E:element 元素
3. K:key 键
4. V:value 值
5. ?:未知类型
示例:
class ArrayAlg {
public static <T> T getMiddle(T... a) {
return a[a.length / 2];
}
}
同一个类的方法,使用不同类型的数组,都可以正常得到数据。
String[] array = {"123","345","456" , "567"};
String string = ArrayAlg.getMiddle(array);
System.out.println( string );
Integer[] array2 = {33,44,55,66,77,88};
System.out.println( ArrayAlg.getMiddle(array2) );
#有限制的通配符
考虑一个简单的画图程序,它可以用来画各种形状,比如矩形和圆形,为了在程序中表示这些形状,你可以定义下面的类继承结构。
// 抽象类
public abstract class Shape {
public abstract void draw();
}
// 画布
public class Canvas {
public void draw(Shape s) {
s.draw();
}
}
/// ---- 抽象类的子类-----------------------
public class Circle extends Shape {
private int x, y, radius;
public void draw() {
// ...
}
}
public class Rectangle extends Shape {
private int x, y, width, height;
public void draw() {
// ...
}
}
所有的图形通常都有很多个形状。假定它们用一个 list 来表示,Canvas 里有一个方法来画出所有的形状会比较方便.
import java.util.List;
public class Canvas {
public void draw(Shape s) {
s.draw();
}
public void drawAll(List<Shape> shapes) {
for (Shape s : shapes) {
s.draw();
}
}
}
现在,类型规则导致 drawAll()只能使用 Shape的list 来调用。它不能,比如说对 List<Circle>来调用。 这很不幸, 因为这个方法所作的只是从这个 list 读取 shape,因此它应该也能对 List<Circle>调用。我们真正要的是这个方法能够接受一个任意种类的 shape.
import java.util.List;
public class Canvas {
public void draw(Shape s) {
s.draw();
}
// 注意方法参数的变化
public void drawAll(List<? extends Shape> shapes) {
for (Shape s : shapes) {
s.draw();
}
}
}
我们把类型 List<Shape> 替换成了 List<? extends Shape>。现在drawAll()可以接受任何 Shape 的子类的 List,所以我们可以对 List<Circle>进行调用
List<? extends Shape>是有限制通配符的一个例子。这里?代表一个未知的类型,就像我们前面看到的通配符一样。但是,在这里,我们知道这个未知的类型实际上是Shape 的一个子类(它可以是 Shape本身或者 Shape 的子类而不必是 extends 自 Shape)。我们说 Shape是这个通配符的上限(upper bound)。
像平常一样,要得到使用通配符的灵活性有些代价。这个代价是,现在向 shapes 中写入是非法的。比如下面的代码是不允许的。
public void addRectangle(List<? extends Shape> shapes) {
// 编译时会报错
shapes.add( new Rectangle());
}
shapes.add 的第二个参数类型是? extends Shape ——一个 Shape 未知的子类。因此我们不知道这个类型是什么,我们不知道它是不是 Rectangle 的父类;它可能是也可能不是一个父类,所以这里传递一个 Rectangle 不安全
擦除和翻译
public static String loophole(Integer x) {
List<String> ys = new LinkedList<String>();
List xs = ys;
xs.add(x);
return ys.iterator().next();
}
public static void main(String[] args) {
loophole(123);
}