Java泛型解析(03):虚拟机运行泛型代码

     Java虚拟机是不存在泛型类型对象的,全部的对象都属于普通类,甚至在泛型实现的早起版本号中,可以将使用泛型的程序编译为在1.0虚拟机上可以执行的class文件,这个向后兼容性后期被抛弃了,所以后来假设用Sun公司的编译器编译的泛型代码,是不能执行在Java5.0之前的虚拟机的,这样就导致了一些实际生产的问题,如一些遗留代码怎样跟新的系统进行衔接,要弄明确这个问题,须要先了解一下虚拟机是怎么执行泛型代码的。
       虚拟机的一种机制:擦除类型參数,并将其替换成特定类型,没有指定特定类型用Object取代,如前文中的Couple<T>类,虚拟机擦除后:   
[code01]

     public class Couple {
private Object wife ;
private Object husband ; public Couple(Object wife, Object husband) {
this.wife = wife;
this.husband = husband;
}
public void setWife(Object wife) {this. wife = wife;}
public void setHusband(Object husband) {this. husband = husband;} public Object getWife() {return wife;}
public Object getHusband() {return husband;}
}
     类型參数T是一个随意类型的,所以擦除后用Object取代了。无论是Couple<Employee>或者Couple<String>擦除后都成为了原始类Couple类,这就好比回到了泛型引入Java之前的普通类。所以这里重点环绕着擦除类型參数这个机制展开解说。
     如有对类型參数有类型限定会怎么替换呢?擦除类型參数机制告诉我们,使用限定的类型取代,假设有多个,使用第一个取代,看一段代码:
[code02]

     public class Period<T extends Comparable<T> & Serializable> {
private T begin;
private T end; public Period(T one, T two) {
if (one.compareTo(two) > 0) {begin = two;end = one;
} else {begin = one;end = two;}
}
}
     code02擦除后,Period的原始类型例如以下:
[code03]

     public class Period {
private Comparable begin;
private Comparable end; public Period(Comparable one, Comparable two) {
if (one.compareTo(two) > 0) {begin = two; end = one;
} else {begin = one; end = two;}
}
}
     思考一下,假设将Period<T extends Comparable<T> & Serializable>写成Period<T extends Serializable 
& Comparable<T>>会是怎么样呢?同理,擦除后原始类型用第一个Serializable取代,这样进行compareTo方法调用的时候,编译器会进行必要的强制类型转换,所以为了提高效率,将标签接口(没有不论什么方法的接口,也叫tagging接口)放在后面。
     先来看看虚拟机运行表达式的时候发生了什么,如:
[code04]

     Couple<Employee> couple = ...;
Employee wife = couple.getWife();
     擦除后,getWife()返回的是Object类型,然后虚拟机会插入强制类型转换,将Object转换为Employee,所以虚拟机实际上运行了两天指令:
     1.调用Couple.getWife()方法。
     2.将Object转换成Employee类型。
     再来看看虚拟机运行泛型方法的时候发生了什么,泛型方法如:
[code05]

public static <T extends Comparable<T>> max(T[] arrays) {... }
擦除后成了:
public staticComoparable max(Comparable[] arrays) {... }
     可是泛型方法的擦除会带来两个复杂的问题,且看第一个实例,一个实例:
[code06]

     public class Period <T extends Comparable<T> & Serializable> {
private T begin;
private T end; public Period(T one, T two) {
if (one.compareTo(two) > 0) {begin = two;end = one;
} else {begin = one;end = two;}
}
public void setBegin(T begin) {this. begin = begin;}
public void setEnd(T end) {this. end = end;}
public T getBegin() {return begin;}
public T getEnd() {return end;}
}
public class DateInterval extends Period<Date> { public DateInterval(Date one, Date two) {
super(one, two);
}
public void setBegin(Date begin) {
super.setBegin(begin);
}
}
     DateInterval类型擦除后,Period中的方法变成:
     public void setBegin(Object begin)
{...}
     而DateInterval中的方法还是:
     public void setBegin(Date
begin) {...}
     所以DateInterval从Period中继承了 public void setBegin(Object begin)
{...}而自身又存在public void setBegin(Date begin) {...}方法,用户使用时问题发生了:
[code07]

     Period<Date> period  = new DateInterval(...);
period.setBegin(new Date());
     这里由于period引用指向了DateInterval实例,依据多态性,setBegin应该调用DateInterval对象的setBegin方法,但是这个擦除让Period中的 public void setBegin(Object begin)
{...}被调用,导致了擦除与多态发生了冲突,怎么办呢?虚拟机此时会在DateInterval类中生成一个桥方法(bridge method),调用过程发生了细微的变化:
[code08]

     public void setBegin(Object begin) {
setBegin((Date)begin);
}
     有了这个合成的桥方法以后,code07中对setBegin的调用过程例如以下:
      1.调用DateInterval.setBegin(Object)方法。
      2.DateInterval.setBegin(Object)方法调用DateInterval.setBegin(Date)方法。
     发现了吗,当我们在DateInterval中添加了getBegin方法之后会是什么样子的呢?是不是Peroid中有一个Object getBegin()的方法,而DateInterval中有一个Date getBegin()方法呢,这两个方法在Java中是不能同一时候存在的?但是Java5以后添加了一个协变类型,使得这里是被同意的,看看DateInterval中getBegin方法就知道了:
[code09]

     @Override
public Date getBegin(){ return super.getBegin(); }
     这里用了@Override,说明是覆盖了父类的Object getBegin()方法,而返回值能够指定为父类中的返回值类型的子类,这就是协变类型,这是Java5以后才干够同意的,同意子类覆盖了方法后指定一个更严格的类型(子类型)。
总结:
     1.记住一点,虚拟机中没有泛型,仅仅有普通的类。
     2.全部泛型的类型參数都用它们限定的类型取代,没有限定则用Object。
     3.为了保持类型安全性,虚拟机在有必要时插入强制类型转换。
     4.桥方法的合成用来保持多态性。
     5.协变类型同意子类覆盖方法后返回一个更严格的类型。


=====【感谢亲阅读寻常心的文章,亲若认为此文有帮助,顶一顶亲的支持将给我前进的动力】=====

Java泛型解析(03):虚拟机运行泛型代码的更多相关文章

  1. Java泛型解析(01):认识泛型

    Java泛型解析(01):认识泛型 What      Java从1.0版本号到如今的8.中间Java5中发生了一个非常重要的变化,那就是泛型机制的引入.Java5引入了泛型,主要还是为了满足在199 ...

  2. Java:IDEA设置虚拟机运行时参数

    第一步:打开“Run->Edit Configurations”菜单 第二步:选择“VM Options”选项,输入你要设置的VM参数 第三步:点击“OK”.“Apply”后设置完成

  3. Java泛型解析(04):约束和局限性

    Java泛型解析(04):约束和局限性           前两节.认识和学习了泛型的限定以及通配符.刚開始学习的人可能须要一些时间去体会到泛型程序设计的优点和力量,特别是想成为库程序猿的同学就须要下 ...

  4. Java泛型解析(02):通配符限定

    Java泛型解析(02):通配符限定      考虑一个这种场景.计算数组中的最大元素. [code01] public class ArrayUtil { public static <T&g ...

  5. WCF 已知类型和泛型解析程序 KnownType

    数据协定继承 已知类型和泛型解析程序 Juval Lowy 下载代码示例 自首次发布以来,Windows Communication Foundation (WCF) 开发人员便必须处理数据协定继承方 ...

  6. java基础(9) - 泛型解析

    泛型 定义简单的泛型类 泛型方法 /** * 1.定义一个泛型类 * 在类名后添加类的泛型参数 <T> * 泛型类里面的所有T会根据创建泛型类时传入的参数确定类型 * 2.定义泛型方法 * ...

  7. 《徐徐道来话Java》(2):泛型和数组,以及Java是如何实现泛型的

    数组和泛型容器有什么区别 要区分数组和泛型容器的功能,这里先要理解三个概念:协变性(covariance).逆变性(contravariance)和无关性(invariant). 若类A是类B的子类, ...

  8. 从fastjson多层泛型嵌套解析,看jdk泛型推断

    给你一组json数据结构,你把它解析出来到项目中,你会怎么做? // data1 sample { "code" : "1", "msg" ...

  9. java泛型(二)、泛型的内部原理:类型擦除以及类型擦除带来的问题

    微信公众号[程序员江湖] 作者黄小斜,斜杠青年,某985硕士,阿里 Java 研发工程师,于 2018 年秋招拿到 BAT 头条.网易.滴滴等 8 个大厂 offer,目前致力于分享这几年的学习经验. ...

随机推荐

  1. UVa 11069 - A Graph Problem

    题目:给你一个集合{1,2,..,n},计算子集的个数,子集的元素不能相邻且不能再插入元素. 分析:dp,动态规划.相邻元素间仅仅能相差3或者2. 动态方程:f(k)= f(k-2)+ f(k-3): ...

  2. 接近带给你AngularJS - 经验说明示例

    接近带给你AngularJS列: 带你走近AngularJS - 基本功能介绍 带你走近AngularJS - 体验指令实例 带你走近AngularJS - 创建自己定义指令 ------------ ...

  3. Source insight 3572安装和版本号An invalid source insight serial number was detected解

    Source insight最新版本3572 下载链接:http://www.sourceinsight.com/down35.html,   http://www.sourceinsight.com ...

  4. 【转】Android 4.3源码下载及问题解决

    [html] view plaincopy 1 2 3 4 5 6 7 8 9 10 11 jianguoliao@jianguoliao-Lenovo-IdeaPad-Y470:~$ cat /et ...

  5. Windowsport80解决方案被占用

    今天,在一个非常沮丧的实施Server什么时候,一个错误port80占用.因此,找到一种方法来解决各类.最后,我的解决方案列出的问题来,要遇到的人做一些参考同样的问题. 第一步,找出哪些程序正在使用p ...

  6. 跨域请求jQuery的ajax jsonp使用常见问题解答

    前天在项目中写了ajax jsonp的使用,出现了问题:能够成功获得请求结果,但没有运行success方法,直接运行了error方法提示错误--ajax jsonp之前并没实用过.对其的理解为跟普通的 ...

  7. Hadoop Hive sql 语法详细解释

    Hive 是基于Hadoop 构建的一套数据仓库分析系统.它提供了丰富的SQL查询方式来分析存储在Hadoop 分布式文件系统中的数据,能够将结构 化的数据文件映射为一张数据库表,并提供完整的SQL查 ...

  8. IT该忍者神龟Jquery小工具easyUI物业摘要召回

    找了个时间看了下EasyUI插件.对它的插件感觉是非常舒服,特地把Easy UI的大部分功能属性做了一下汇总. 此属性列表请对比jQuery EasyUI 1.0.5,关于它的很多其它资讯请猛击这里. ...

  9. 【C++基金会 04】vector详细解释

    根据写博客开始总有一些事情的习惯,加鸡汤文,今天请原谅我记得. ============================================= 今天要写的内容是顺序型容器.首先,标准库定义 ...

  10. 从源代码分析modelDriven拦截器和params拦截器和拦截器prepare 和paramsPrepareParamsStack拦截器栈(让你的Struts2代码更简洁——如何培养框架设计能力

    源代码文件:Web App Libraries/struts2-core-2.3.15.3.jar/struts-default.xml 拦截器modelDriven: <interceptor ...