转载:http://www.iteye.com/topic/549509
- //泛型代码
- public class Pair<T>{
- private T first= null ;
- private T second= null ;
- public Pair(T fir,T sec){
- this .first=fir;
- this .second=sec;
- }
- public T getFirst(){
- return this .first;
- }
- public T getSecond(){
- return this .second;
- }
- public void setFirst(T fir){
- this .first=fir;
- }
- }
上面是一个很典型的泛型(generic)代码。T是类型变量,可以是任何引用类型。
1、Generic class 创建对象
Pair<String> pair1=new Pair("string",1); ...①
Pair<String> pair2=new Pair<String>("string",1) ...②
有个很有趣的现象:
①代码在编译期不会出错,②代码在编译期会检查出错误。
这个问题其实很简单
(1) JVM本身并没有泛型对象这样的一个特殊概念。所有的泛型类对象在编译器会全部变成普通类对象(这一点会在下面详细阐述)。
比如①,②两个代码编译器全部调用的是 Pair(Object fir, Object sec)这样的构造器。
因此代码①中的new Pair("string",1)在编译器是没有问题的,毕竟编译器并不知道你创建的Pair类型中具体是哪一个类型变量T,而且编译器肯定了String对象和Integer对象都属于Object类型的。
但是一段运行pair1.getSecond()就会抛出ClassCastException异常。这是因为JVM会根据第一个参数"string"推
算出T类型变量是String类型,这样getSecond也应该是返回String类型,然后编译器已经默认了second的操作数是一个值为1的
Integer类型。当然就不符合JVM的运行要求了,不终止程序才怪。
(2) 但代码②会在编译器报错,是因为new Pair<String>("string",1)已经指明了创建对象pair2的类型变量T应该是String的。所以在编译期编译器就知道错误出在第二个参数Integer了。
小结一下:
创建泛型对象的时候,一定要指出类型变量T的具体类型。争取让编译器检查出错误,而不是留给JVM运行的时候抛出异常。
2、JVM如何理解泛型概念
—— 类型擦除
事实上,JVM并不知道泛型,所有的泛型在编译阶段就已经被处理成了普通类和方法。
处理方法很简单,我们叫做类型变量T的擦除(erased)
。
无论我们如何定义一个泛型类型,相应的都会有一个原始类型被自动提供。原始类型的名字就是擦除类型参数的泛型类型的名字。
如果泛型类型的类型变量没有限定(<T>)
,那么我们就用Object作为原始类型;
如果有限定(<T extends XClass>),我们就XClass作为原始类型;
如果有多个限定(<T extends XClass1&XClass2>),我们就用第一个边界的类型变量XClass1类作为原始类型;
比如上面的Pair<T>例子,编译器会把它当成被Object原始类型替代的普通类来替代。
- //编译阶段:类型变量的擦除
- ublic class Pair{
- private Object first= null ;
- private Object second= null ;
- public Pair(Object fir,Object sec){
- this .first=fir;
- this .second=sec;
- }
- public Object getFirst(){
- return this .first;
- }
- public void setFirst(Object fir){
- this .first=fir;
- }
- }
3、泛型约束和局限性—— 类型擦除所带来的麻烦
(1) 继承泛型类型的多态麻烦。(—— 子类没有覆盖住父类的方法 )
看看下面这个类SonPair
很明显,程序员的本意是想在SonPair类中覆盖父类Pair<String>的setFirst(T fir)这个方法。但事实上,SonPair中的setFirst(String fir)方法根本没有覆盖住Pair<String>中的这个方法。
原因很简单,Pair<String>在编译阶段已经被类型擦除为Pair了,它的setFirst方法变成了setFirst(Object fir)。
那么SonPair中
setFirst(String)当然无法覆盖住父类的setFirst(Object)了。
这对于多态来说确实是个不小的麻烦,我们看看编译器是如何解决这个问题的。
编译器
会自动在
SonPair中生成一个桥方法(bridge method
)
:
public void setFirst(Object fir){
setFirst((String) fir)
}
这样,SonPair的桥方法确实能够覆盖泛型父类的setFirst(Object)
了。而且桥方法内部其实调用的是子类字节setFirst(String)方法。对于多态来说就没问题了。
问题还没有完,多态中的方法覆盖是可以了,但是桥方法却带来了一个疑问:
现在,假设 我们还想在 SonPair 中覆盖getFirst()方法呢?
由于需要桥方法来覆盖父类中的getFirst,编译器会自动在SonPair中生成一个 public Object getFirst()桥方法。
但是,疑问来了,SonPair中出现了两个方法签名一样的方法(只是返回类型不同):
①String getFirst() // 自己定义的方法
②Object getFirst() // 编译器生成的桥方法
难道,编译器允许出现方法签名相同的多个方法存在于一个类中吗?
事实上有一个知识点可能大家都不知道:
① 方法签名
确实只有方法名+参数列表
。这毫无疑问!
② 我们绝对不能编写出方法签名一样的多个方法
。如果这样写程序,编译器是不会放过的。这也毫无疑问!
③ 最重要的一点是:JVM会用参数类型和返回类型来确定一个方法。
一旦编译器通过某种方式自己编译出方法签名一样的两个方法(只能编译器自己来创造这种奇迹,我们程序员却不能人为的编写这种代码)。JVM还是能够分清楚这些方法的,前提是需要返回类型不一样。
(2) 泛型类型中的方法冲突
还是来看一段代码:
- //在上面代码中加入equals方法
- public class Pair<T>{
- public boolean equals(T value){
- return (first.equals(value));
- }
- }
这样看似乎没有问题的代码连编译器都通过不了:
【Error】 Name clash: The method equals(T) of type Pair<T> has the same erasure as equals(Object) of type Object but does not override it。
编译器说你的方法与Object中的方法冲突了。这是为什么?
开始我也不太明白这个问题,觉得好像编译器帮助我们使得equals(T)这样的方法覆盖上了Object中的equals(Object)。经过大家的讨论,我觉得应该这么解释这个问题?
首先、我们都知道子类方法要覆盖,必须与父类方法具有相同的方法签名(方法名+参数列表)。而且必须保证子类的访问权限>=父类的访问权限。这是大家都知道的事实。
然后、在上面的代码中,当编译器看到Pair<T>中的equals(T)方法时,第一反应当然是equals(T)没有覆盖住父类Object中的equals(Object)了。
接着、编译器将泛型代码中的T用Object替代(擦除)。突然发现擦除以后equals(T)变成了equals(Object),糟糕了,这个方法与 Object类中的equals一样了。基于开始确定没有覆盖这样一个想法,编译器彻底的疯了(精神分裂)。然后得出两个结论:①坚持原来的思想:没有覆 盖。但现在一样造成了方法冲突了。 ②写这程序的程序员疯了(哈哈)。
再说了,拿Pair<T>对象和T对象比较equals,就像牛头对比马嘴,哈哈,逻辑上也不通呀。
(3) 没有泛型数组一说
Pair<String>[] stringPairs=new Pair<String>[10];
Pair<Integer>[] intPairs=new Pair<Integer>[10];
这种写法编译器会指定一个Cannot create a generic array of Pair<String>的错误
我们说过泛型擦除之后,Pair<String>[]会变成Pair[],进而又可以转换为Object[];
假设泛型数组存在,那么
Object[0]=stringPairs[0]; Ok
Object[1]=intPairs[0]; Ok
这就麻烦了,理论上将Object[]可以存储所有Pair对象,但这些Pair对象是泛型对象,他们的类型变量都不一样,那么调用每一个 Object[]数组元素的对象方法可能都会得到不同的记过,也许是个字符串,也许是整形,这对于JVM可是无法预料的。
记住: 数组必须牢记它的元素类型,也就是所有的元素对象都必须一个样,泛型类型恰恰做不到这一点。即使Pair<String>,Pair<Integer>... 都是Pair类型的,但他们还是不一样。
总结:泛型代码与JVM
① 虚拟机中没有泛型,只有普通类和方法。
② 在编译阶段,所有泛型类的类型参数都会被Object或者它们的限定边界来替换。(类型擦除)
③ 在继承泛型类型的时候,桥方法的合成是为了避免类型变量擦除所带来的多态灾难。
发表评论
-
(转载)又见内存泄露
2012-02-27 20:54 0转载:http://www.iteye.com/topic/1 ... -
(转载)详解spring事务属性
2012-02-27 20:53 0转载:http://www.iteye.com/t ... -
(转载)简明扼要,再谈ThreadLocal和synchronized
2012-02-27 20:52 0转载:http://www.iteye.com/topic/1 ... -
(转载)Spring声明式事务管理源码解读之事务开始
2012-02-27 20:51 0转载:http://www.iteye.com/topic/8 ... -
(转载)Web Services开发体会和项目教训
2012-02-27 20:50 0转载:http://www.iteye.com/topic/7 ... -
(转载)使用JDBC和Hibernate来写入Blob型数据到Oracle中
2012-02-27 20:49 0转载:http://www.iteye.com/topic/2 ... -
(转载)Java学习从入门到精通
2012-02-27 20:45 0转载:http://www.iteye.com/t ... -
(转载)解惑 spring 嵌套事务
2012-02-27 20:43 0转载:http://www.iteye.com/topic/3 ... -
(转载)Spring源代码解析(一):IOC容器
2012-02-27 20:41 0转载:http://www.iteye.com/topic/8 ... -
(转载)Word/Excel/PDF文件转换成HTML整理
2012-02-26 23:24 0转载:http://www.iteye.com/topic/5 ... -
(转载)为什么需要Singleton
2012-02-26 23:22 0转载:http://www.iteye.com/topic/4 ... -
(转载)java并发编程-Executor框架
2012-02-26 23:21 0转载:http://www.iteye.com/topic/3 ... -
(转载)Spring注解入门
2012-02-26 23:21 0转载:http://www.iteye.com/topic/2 ... -
(转载)一个简单例子:贫血模型or领域模型
2012-02-26 23:19 714转载:http://www.iteye.com/topic/2 ... -
(转载)String类的split方法引起的内存泄漏
2012-02-26 23:14 957转载:原文地址:http://jarfield.iteye.c ...
相关推荐
### JVM如何理解Java泛型类 #### 一、引言 在Java中,泛型是一种强大的功能,它允许程序员编写灵活且类型安全的代码。然而,对于Java虚拟机(JVM)来说,它实际上并不理解泛型的概念。所有的泛型信息在编译阶段就被...
Java泛型深入的内容涵盖泛型的基本概念、泛型类、接口、方法以及泛型的使用限制和高级特性。 首先,Java中的泛型允许定义方法、接口、类和变量时不指定具体的数据类型,而是在使用的时候再通过泛型类型参数来指定...
Java泛型可以用于各种场景,例如集合、类、接口等。例如,定义一个List<String> strList = new ArrayList(),那么strList只能添加String类型的数据。 泛型的优点 ------------- Java泛型的优点是可以在编译期检查...
让我们深入探讨一下Java泛型的各个方面。 首先,泛型(Generic type)是一种在定义类、接口或方法时,使用类型参数的方式。类型参数就像是方法的形参,但用于类型而非值。例如,当我们创建一个泛型列表`List<T>`,`...
Java泛型的一个重要应用是在集合框架中。在JDK 1.5之前,Java集合类框架使用Object作为元素类型,这意味着集合可以存储任何类型的对象。然而,这种设计导致在从集合中获取元素时必须进行类型转换,这个过程是繁琐且...
Java 泛型是一种强大的语言特性,它允许在类、接口和方法中使用类型参数,以实现类型的安全性和代码的重用性。泛型是 Java 从 JDK 5.0 版本开始引入的重要特性,目的是增强类型系统,提高程序的类型安全性,减少运行...
1. Java泛型是编译时特性,在JVM运行时不会保留泛型的类型信息。 2. Java泛型利用编译器擦除技术实现,这意味着泛型的源码在编译后会转化为非泛型版本的源码。 3. 泛型的类型擦除指的是在编译器前端处理阶段,去除...
Java泛型的实现也带来了一些特性,比如泛型类没有自己独立的Class对象,所有的泛型类都共享一个Class对象,比如List.class。静态变量在泛型类的所有实例中是共享的,不论创建这些实例时使用的类型参数是什么。这意味...
3. **潜在的性能收益**:虽然Java泛型的初始实现并未直接提升运行时性能,但其为编译器提供了更多的类型信息,为未来JVM的优化提供了可能性。例如,编译器可能能够更有效地处理泛型代码,减少运行时的检查。 4. **...
根据提供的文件信息,我们可以确定本书的标题为《Java泛型和集合》(Java Generics and Collections),作者为Maurice Naftalin和Philip Wadler。该书详细介绍了Java中的泛型(Generics)特性以及集合(Collections...
System 类代表了当前 JVM 虚拟机对应的操作系统对象,可以拿系统时间、退出 JVM 虚拟机、复制数组等。 StringBuilder 类是一个字符串缓冲区,它可以用来进行字符串的拼接、删除、反转等操作。它比 String 类更高效...
### Java泛型梳理 #### 一、泛型概念解析 泛型是一种参数化的类型机制,其核心在于将具体的类型参数化处理。这种处理方式类似于函数中的变量参数——具体类型可以在定义时留空,而在实际使用时传入。这种方式极大...
Java 泛型是一种强大的特性,它...总之,Java泛型是一个强大的工具,它可以提高代码的类型安全性,减少错误,使代码更易于理解和维护。在编写Java程序时,尤其是在处理集合和数据结构时,应该尽可能地利用泛型的优势。
Java泛型是Java编程语言中的一个关键特性,它允许在定义类、接口和方法时指定类型参数,从而增强了代码的类型安全性和重用性。在本小测试中,我们将探讨几个与Java泛型相关的概念:通配符(Wildcards)、消除...
标题和描述均聚焦于“基于Java的泛型编程”,这一主题深入探讨了Java语言中泛型编程的概念、优点以及其实现机制。...理解并熟练掌握泛型编程,对于任何希望在Java开发中取得卓越成就的程序员来说,都是至关重要的技能。
本项目“jvmjava”是一个开源项目,由Java语言实现,目的是为了让开发者能够更深入地理解JVM的工作原理,并提供了一个动手实践的机会。 一、JVM概述 Java虚拟机(JVM)是Java平台的核心,它负责执行字节码,提供了...