ImageVerifierCode 换一换
格式:DOCX , 页数:11 ,大小:62.65KB ,
资源ID:1071616      下载积分:1 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.bingdoc.com/d-1071616.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(Java泛型简明教程.docx)为本站会员(b****2)主动上传,冰点文库仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知冰点文库(发送邮件至service@bingdoc.com或直接QQ联系客服),我们立即给予删除!

Java泛型简明教程.docx

1、Java泛型简明教程导读:本文是从Java Generics Quick Tutorial这篇文章翻译而来,译文来自外刊IT评论Java泛型简明教程。内容如下:泛型是Java SE 5.0中引入的一项特征,自从这项语言特征出现多年来,我相信,几乎所有的Java程序员不仅听说过,而且使用过它。关于Java泛型的教程,免费的,不免费的,有很多。我遇到的最好的教材有:The Java TutorialJava Generics and Collections, by Maurice Naftalin and Philip WadlerEffective Java中文版(第2版), by Joshua

2、 Bloch.尽管有这么多丰富的资料,有时我感觉,有很多的程序员仍然不太明白Java泛型的功能和意义。这就是为什么我想使用一种最简单的形式来总结一下程序员需要知道的关于Java泛型的最基本的知识。Java泛型由来的动机理解Java泛型最简单的方法是把它看成一种便捷语法,能节省你某些Java类型转换(casting)上的操作:1. Listbox=.; 2. Appleapple=box.get(0); 上面的代码自身已表达的很清楚:box是一个装有Apple对象的List。get方法返回一个Apple对象实例,这个过程不需要进行类型转换。没有泛型,上面的代码需要写成这样:1. Listbox=

3、.; 2. Appleapple=(Apple)box.get(0); 很明显,泛型的主要好处就是让编译器保留参数的类型信息,执行类型检查,执行类型转换操作:编译器保证了这些类型转换的绝对无误。相对于依赖程序员来记住对象类型、执行类型转换这会导致程序运行时的失败,很难调试和解决,而编译器能够帮助程序员在编译时强制进行大量的类型检查,发现其中的错误。泛型的构成由泛型的构成引出了一个类型变量的概念。根据Java语言规范,类型变量是一种没有限制的标志符,产生于以下几种情况:泛型类声明泛型接口声明泛型方法声明泛型构造器(constructor)声明泛型类和接口如果一个类或接口上有一个或多个类型变量,那

4、它就是泛型。类型变量由尖括号界定,放在类或接口名的后面:1. publicinterfaceListextendsCollection 2. 3. . 4. 简单的说,类型变量扮演的角色就如同一个参数,它提供给编译器用来类型检查的信息。Java类库里的很多类,例如整个Collection框架都做了泛型化的修改。例如,我们在上面的第一段代码里用到的List接口就是一个泛型类。在那段代码里,box是一个List对象,它是一个带有一个Apple类型变量的List接口的类实现的实例。编译器使用这个类型变量参数在get方法被调用、返回一个Apple对象时自动对其进行类型转换。实际上,这新出现的泛型标记,

5、或者说这个List接口里的get方法是这样的:1. Tget(intindex); get方法实际返回的是一个类型为T的对象,T是在List声明中的类型变量。泛型方法和构造器(Constructor)非常的相似,如果方法和构造器上声明了一个或多个类型变量,它们也可以泛型化。1. publicstaticTgetFirst(Listlist) 这个方法将会接受一个List类型的参数,返回一个T类型的对象。例子你既可以使用Java类库里提供的泛型类,也可以使用自己的泛型类。类型安全的写入数据下面的这段代码是个例子,我们创建了一个List实例,然后装入一些数据:1. Liststr=newArray

6、List(); 2. 3. str.add(Hello); 4. 5. str.add(World.); 如果我们试图在List装入另外一种对象,编译器就会提示错误:1. str.add(1);/不能编译 类型安全的读取数据当我们在使用List对象时,它总能保证我们得到的是一个String对象:1. StringmyString=str.get(0); 遍历类库中的很多类,诸如Iterator,功能都有所增强,被泛型化。List接口里的 iterator()方法现在返回的是Iterator,由它的T next()方法返回的对象不需要再进行类型转换,你直接得到正确的类型。1. for(Itera

7、toriter=str.iterator();iter.hasNext();) 2. 3. Strings=iter.next(); 4. 5. System.out.print(s); 6. 7. 使用foreach“for each”语法同样受益于泛型。前面的代码可以写出这样:1. for(Strings:str) 2. 3. System.out.print(s); 4. 5. 这样既容易阅读也容易维护。自动封装(Autoboxing)和自动拆封(Autounboxing)在使用Java泛型时,autoboxing/autounboxing这两个特征会被自动的用到,就像下面的这段代码:1

8、. Listints=newArrayList(); 2. 3. ints.add(0); 4. 5. ints.add(1); 6. 7. 8. 9. 10. intsum=0; 11. 12. for(inti:ints) 13. 14. sum+=i; 15. 16. 然而,你要明白的一点是,封装和解封会带来性能上的损失,所有,通用要谨慎的使用。子类型在Java中,跟其它具有面向对象类型的语言一样,类型的层级可以被设计成这样:在Java中,类型T的子类型既可以是类型T的一个扩展,也可以是类型T的一个直接或非直接实现(如果T是一个接口的话)。因为“成为某类型的子类型”是一个具有传递性质的关

9、系,如果类型A是B的一个子类型,B是C的子类型,那么A也是C的子类型。在上面的图中:FujiApple(富士苹果)是Apple的子类型Apple是Fruit(水果)的子类型FujiApple(富士苹果)是Fruit(水果)的子类型所有Java类型都是Object类型的子类型。B类型的任何一个子类型A都可以被赋给一个类型B的声明:1. Applea=.; 2. Fruitf=a; 泛型类型的子类型如果一个Apple对象的实例可以被赋给一个Fruit对象的声明,就像上面看到的,那么,List 和 a List之间又是个什么关系呢?更通用些,如果类型A是类型B的子类型,那C 和 C之间是什么关系?答

10、案会出乎你的意料:没有任何关系。用更通俗的话,泛型类型跟其是否子类型没有任何关系。这意味着下面的这段代码是无效的:1. Listapples=.; 2. Listfruits=apples; 下面的同样也不允许:List apples;1. Listfruits=.; 2. apples=fruits; 为什么?一个苹果是一个水果,为什么一箱苹果不能是一箱水果?在某些事情上,这种说法可以成立,但在类型(类)封装的状态和操作上不成立。如果把一箱苹果当成一箱水果会发生什么情况?1. Listapples=.; 2. Listfruits=apples; 3. fruits.add(newStraw

11、berry(); 如果可以这样的话,我们就可以在list里装入各种不同的水果子类型,这是绝对不允许的。另外一种方式会让你有更直观的理解:一箱水果不是一箱苹果,因为它有可能是一箱另外一种水果,比如草莓(子类型)。这是一个需要注意的问题吗?应该不是个大问题。而程序员对此感到意外的最大原因是数组和泛型类型上用法的不一致。对于泛型类型,它们和类型的子类型之间是没什么关系的。而对于数组,它们和子类型是相关的:如果类型A是类型B的子类型,那么A是B的子类型:1. 1Appleapples=.; 2. 2Fruitfruits=apples; 可是稍等一下!如果我们把前面的那个议论中暴露出的问题放在这里,我

12、们仍然能够在一个apple类型的数组中加入strawberrie(草莓)对象:1. Appleapples=newApple1; 2. Fruitfruits=apples; 3. fruits0=newStrawberry(); 这样写真的可以编译,但是在运行时抛出ArrayStoreException异常。因为数组的这特点,在存储数据的操作上,Java运行时需要检查类型的兼容性。这种检查,很显然,会带来一定的性能问题,你需要明白这一点。重申一下,泛型使用起来更安全,能“纠正”Java数组中这种类型上的缺陷。现在估计你会感到很奇怪,为什么在数组上会有这种类型和子类型的关系,我来给你一个Jav

13、a Generics and Collections这本书上给出的答案:如果它们不相关,你就没有办法把一个未知类型的对象数组传入一个方法里(不经过每次都封装成Object),就像下面的:1. voidsort(Objecto); 泛型出现后,数组的这个个性已经不再有使用上的必要了(下面一部分我们会谈到这个),实际上是应该避免使用。通配符在本文的前面的部分里已经说过了泛型类型的子类型的不相关性。但有些时候,我们希望能够像使用普通类型那样使用泛型类型:向上造型一个泛型对象的引用向下造型一个泛型对象的引用向上造型一个泛型对象的引用例如,假设我们有很多箱子,每个箱子里都装有不同的水果,我们需要找到一种

14、方法能够通用的处理任何一箱水果。更通俗的说法,A是B的子类型,我们需要找到一种方法能够将C类型的实例赋给一个C类型的声明。为了完成这种操作,我们需要使用带有通配符的扩展声明,就像下面的例子里那样:1. Listapples=newArrayList(); 2. Listfruits=apples; “? extends”是泛型类型的子类型相关性成为现实:Apple是Fruit的子类型,List 是 List 的子类型。向下造型一个泛型对象的引用现在我来介绍另外一种通配符:? super。如果类型B是类型A的超类型(父类型),那么C 是 C 的子类型:1. Listfruits=newArray

15、List(); 2. List=fruits; 为什么使用通配符标记能行得通?原理现在已经很明白:我们如何利用这种新的语法结构?? extends让我们重新看看这第二部分使用的一个例子,其中谈到了Java数组的子类型相关性:1. Appleapples=newApple1; 2. Fruitfruits=apples; 3. fruits0=newStrawberry(); 就像我们看到的,当你往一个声明为Fruit数组的Apple对象数组里加入Strawberry对象后,代码可以编译,但在运行时抛出异常。现在我们可以使用通配符把相关的代码转换成泛型:因为Apple是Fruit的一个子类,我们

16、使用? extends 通配符,这样就能将一个List对象的定义赋到一个List的声明上:1. Listapples=newArrayList(); 2. Listfruits=apples; 3. fruits.add(newStrawberry(); 这次,代码就编译不过去了!Java编译器会阻止你往一个Fruit list里加入strawberry。在编译时我们就能检测到错误,在运行时就不需要进行检查来确保往列表里加入不兼容的类型了。即使你往list里加入Fruit对象也不行:1. fruits.add(newFruit(); 你没有办法做到这些。事实上你不能够往一个使用了? exten

17、ds的数据结构里写入任何的值。原因非常的简单,你可以这样想:这个? extends T 通配符告诉编译器我们在处理一个类型T的子类型,但我们不知道这个子类型究竟是什么。因为没法确定,为了保证类型安全,我们就不允许往里面加入任何这种类型的数据。另一方面,因为我们知道,不论它是什么类型,它总是类型T的子类型,当我们在读取数据时,能确保得到的数据是一个T类型的实例:1. Fruitget=fruits.get(0); ? super使用 ? super 通配符一般是什么情况?让我们先看看这个:1. Listfruits=newArrayList(); 2. List=fruits; 我们看到frui

18、ts指向的是一个装有Apple的某种超类(supertype)的List。同样的,我们不知道究竟是什么超类,但我们知道 Apple和任何Apple的子类都跟它的类型兼容。既然这个未知的类型即是Apple,也是GreenApple的超类,我们就可以写入:1. fruits.add(newApple(); 2. fruits.add(newGreenApple(); 如果我们想往里面加入Apple的超类,编译器就会警告你:1. fruits.add(newFruit(); 2. fruits.add(newObject(); 因为我们不知道它是怎样的超类,所有这样的实例就不允许加入。从这种形式的类

19、型里获取数据又是怎么样的呢?结果表明,你只能取出Object实例:因为我们不知道超类究竟是什么,编译器唯一能保证的只是它是个Object,因为Object是任何Java类型的超类。存取原则和PECS法则总结 ? extends 和 the ? super 通配符的特征,我们可以得出以下结论:如果你想从一个数据类型里获取数据,使用 ? extends 通配符如果你想把对象写入一个数据结构里,使用 ? super 通配符如果你既想存,又想取,那就别用通配符。这就是Maurice Naftalin在他的Java Generics and Collections这本书中所说的存取原则,以及Joshua Bloch在他的Effective Java这本书中所说的PECS法则。Bloch提醒说,这PECS是指”Producer Extends, Consumer Super”,这个更容易记忆和运用。译文出自:外刊IT评论

copyright@ 2008-2023 冰点文库 网站版权所有

经营许可证编号:鄂ICP备19020893号-2