Skip to content

Java泛型(三)——桥接

xiaofeiyang95 edited this page May 30, 2017 · 1 revision

在认识桥接前,我们先看一个泛型继承的例子:

public class Node<T> {
    public T data;

    public Node(T data) {
        this.data = data;
    }

    public void setData(T data) {
        System.out.println("Node.setData");
        this.data = data;
    }
}
//继承泛型类
public class MyNode extends Node<Integer> {
    public MyNode(Integer data) {
        super(data);
    }

    public void setData(Integer data) {
        System.out.println("MyNode.setData");
        super.setData(data);
    }
}
//测试类
public class Test {
    public static void main(String[] args){
        MyNode mn = new MyNode(5);
        Node n = mn;
        n.setData("Hello");
        Integer x = mn.data;
    }
}

运行结果如下:

Exception in thread "main" java.lang.ClassCastException: java.lang.String cannot be cast to java.lang.Integer
	at fanxing.MyNode.setData(MyNode.java:6)
	at chongxie.Test.main(Test.java:17)

为什么会报出一个类型转换错误呢?我们逐步来进行分析:

  1. 首先我们知道JVM中是不存在泛型的,在运行过程成,会进行擦除,经过擦除后,代码如下:
MyNode mn = new MyNode(5);
Node n = (MyNode)mn;         // A raw type - compiler throws an unchecked warning
n.setData("Hello");
Integer x = (String)mn.data; // Causes a ClassCastException to be thrown.
  1. Node类擦除后的代码如下:
public class Node {

    public Object data;

    public Node(Object data) { this.data = data; }

    public void setData(Object data) {
        System.out.println("Node.setData");
        this.data = data;
    }
}

我们可以看到也就是将泛型转换为Object类型

  1. MyNode类擦除后代码如下:
public class MyNode extends Node {

    public MyNode(Integer data) { super(data); }

     // Bridge method generated by the compiler
    //
    public void setData(Object data) {
        setData((Integer) data);
    }

    public void setData(Integer data) {
        System.out.println("MyNode.setData");
        super.setData(data);
    }
}

这里,我们可以看到这里不仅擦除了泛型,还增加一个setData的方法,其中里面的参数类型为Object类型。实际上, 这里是编译器自动生成的桥接方法!因为擦除后,MyNode类中的setData并没有正确地重写Node类中的setData(因 为我们知道重写是不能改变参数类型的!),因此,为了保证泛型擦除后依旧存在多态性,编译器增加了一个桥接方法 ,以解决泛型的多态性问题。

Java泛型转换的关键

  • 虚拟机没有泛型,只有普通的类和方法
  • 所有的类型参数都用它们的限定类型替换
  • 桥方法被合成来保持多态
  • 为保持类型安全性,必要时插入强制类型转换