热门搜索:  

Java中的类型擦除与桥方法

类型擦除

Java在语法中虽然存在泛型的概念,但是在虚拟机中却没有泛型的概念,虚拟机中所有的类型都是普通类。无论何时定义一个泛型类型,编译后类型会被都被自动转换成一个相应的原始类型。

比如这个类

public class Parent<T>
{
    public void sayHello(T value)
    {
        System.out.println("This is Parent Class, value is " + value);
    }
}

在编译后就变成了

public class Parent
{
    public void sayHello(Object value)
    {
        System.out.println("This is Parent Class, value is " + value);
    }
}

对类型变量进行替换的规则有两条:

  • 若为无限定的类型,如<T>,被替换为Object
  • 若为限定类型,如<T extends Comparable & Serializable>,则用第一个限定的类型变量来替换,在这里被替换为Comparable

桥方法

类型擦除后,就产生了一个奇怪的现象。

假设有一个超类:

public class Parent<T>
{
    public void sayHello(T value)
    {
        System.out.println("This is Parent Class, value is " + value);
    }
}

以及一个子类:

public class Child extends Parent<String>
{
    public void sayHello(String value)
    {
        System.out.println("This is Child class, value is " + value);
    }
}

最后有以下测试代码,企图实现多态:

public class MainApp
{
    public static void main(String[] args)
    {
        Child child = new Child();
        Parent<String> parent = child;

        parent.sayHello("This is a string");
    }
}

运行的时候,会对Child类的方法表进行搜索,先分析一下Child类的方法表里有哪些东西:

1. sayHello(Object value) : 从类型被擦除后的超类中继承过来
2. sayHello(String value) : 自己新增的方法,和超类毫无联系
3. 一些从Object类继承来的方法,这里忽略

按理来说,这段测试代码应该不能通过编译,因为要实现多态的话,所调用的方法必须在子类中重写,但是在这里Child类并没有重写Parent类中的sayHello(Object value)方法,只是单纯的继承而已,并且新加了一个参数不同的同名方法。

但是结果是可以正常运行。

原因是编译器在Child类中自动生成了一个桥方法:

public void sayHello(Object value)
{
    sayHello((String) value);
}

可以看出,这个桥方法实际上就是对超类中sayHello(Obejct)的重写。这样做的原因是,当程序员在子类中写下以下这段代码的时候,本意是对超类中的同名方法进行重写,但因为超类发生了类型擦除,所以实际上并没有重写成功,因此加入了桥方法的机制来避免类型擦除与多态发生冲突。

public class Child extends Parent<String>
{
    public void sayHello(String value)
    {
        System.out.println("This is Child class, value is " + value);
    }
}

桥方法并不需要自己手动生成,一切都是编译器自动完成的。

桥方法与Geter

同样的,如果超类中有getter的话,在使用多态的时候也可能发生冲突。假设有超类被类型擦除后存在这样一个方法:

Obejct getValue()

然后在子类中,程序员想要重写这个方法,因此新增了一个这样的方法:

String getValue()

但是正如前面所述,重写并没有起作用,甚至还应该报错,因为在子类中,根据 函数签名=方法名+参数 的原则,从超类继承的方法与新增的方法冲突了。

但实际上这样的代码是可以工作的,原因在于,JVM是用返回值+方法名+参数的方式来计算函数签名的,所以编译器就可以借助这一原则来生成一个桥方法。不过这种计算函数签名的方法仅仅存在于虚拟机中。

当前文章:http://4gi.promodchallenge.com/a/7196d_63.html

发布时间:2017-10-19 03:19:34

我的心没有回程    我的心没有回程  我的心没有回程  我的心没有回程  我的心没有回程  我的心没有回程  我的心没有回程  我的心没有回程  我的心没有回程  

http://www.hnhx.net.cnhttp://www.hnhx.net.cn/4047934051/201710100688.htmlhttp://www.hnhx.net.cn/F902t5Sp0/201710100139.htmlhttp://www.hnhx.net.cn/543h2WG3x1/201710100090.htmlhttp://www.hnhx.net.cn/5B068958d/201710100369.htmlhttp://www.uknet.cn/zzqq/2017/1015/24072.htmlhttp://www.uknet.cn/lvyou/jingdian/2017/1015/24079.htmlhttp://www.uknet.cn/zzqq/2017/1015/24081.htmlhttp://www.uknet.cn/jiancai/24082.htmlhttp://www.uknet.cn/a/anquanzixun/shoujianquan/2017/1015/24098.htmlhttp://www.uknet.cn/zzqq/2017/1015/24099.htmlhttp://www.uknet.cn/qushi/24100.htmlhttp://www.uknet.cn/lvyou/youji/2017/1015/24110.htmlhttp://www.uknet.cn/difang/yunan/24113.htmlhttp://www.uknet.cn/qushi/24114.htmlhttp://www.uknet.cn/qiche/hangqing/24139.htmlhttp://www.uknet.cn/licai/24144.htmlhttp://www.uknet.cn/zzqq/2017/1016/24155.htmlhttp://www.uknet.cn/zzqq/2017/1016/24161.htmlhttp://www.uknet.cn/licai/24163.htmlhttp://www.uknet.cn/jiaoyu/24177.htmlhttp://www.uknet.cn/zzqq/2017/1017/24180.htmlhttp://www.uknet.cn/licai/24179.htmlhttp://www.uknet.cn/video/2017/1017/24223.htmlhttp://www.uknet.cn/zzqq/2017/1017/24239.htmlhttp://www.uknet.cn/qushi/24260.htmlhttp://www.uknet.cn/zqbf/2017/1018/24263.htmlhttp://www.uknet.cn/licai/24268.htmlhttp://www.uknet.cn/qushi/24274.htmlhttp://www.uknet.cn/zqbf/2017/1018/24275.htmlhttp://www.uknet.cn/jiaoyu/24282.htmlhttp://www.uknet.cn/zzqq/2017/1018/24306.htmlhttp://www.uknet.cn/zqbf/2017/1020/24308.htmlhttp://www.uknet.cn/difang/yuxi/24312.htmlhttp://www.uknet.cn/licai/24316.htmlhttp://www.uknet.cn/video/guoji/2017/1020/24342.htmlhttp://www.uknet.cn/licai/24343.htmlhttp://www.uknet.cn/licai/24346.htmlhttp://www.uknet.cn/qiche/hangqing/24348.htmlhttp://www.uknet.cn/video/guoji/2017/1020/24354.htmlhttp://www.uknet.cn/jiaoyu/24368.htmlhttp://www.uknet.cn/licai/24391.htmlhttp://www.uknet.cn/zzqq/2017/1022/24401.htmlhttp://www.uknet.cn/qushi/24403.htmlhttp://www.uknet.cn/zzqq/2017/1022/24423.htmlhttp://www.uknet.cn/shichang/24429.htmlhttp://www.uknet.cn/licai/24430.htmlhttp://www.uknet.cn/licai/24437.htmlhttp://www.uknet.cn/qushi/24114.htmlhttp://www.uknet.cn/licai/24346.html