`
carrot
  • 浏览: 163799 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

轻松掌握 Java 泛型

阅读更多

诊断 Java 代码: 轻松掌握 Java 泛型

http://www.ibm.com/developerworks/cn/java/j-djc02113/

Java Tiger 版本和 JSR-14 原型编译器中的泛型指南

 
 
 
  
  

未显示需要 JavaScript 的文档选项



级别: 初级

Eric E. Allen (eallen@cs.rice.edu), 博士研究生, Rice 大学 Java 编程语言团队

2003 年 5 月 14 日

本月的 诊断 Java 代码介绍泛型类型(generic type)和支持它们的特性,计划在 2003 年末发布的 Tiger,也就是 Java V1.5 中打算包含这些泛型和特性。Eric Allen 提供了代码样本,这些样本通过重点描述诸如基本类型的限制、受限泛型和多态方法之类的 Tiger 特性来说明泛型类型的优缺点(即将发表的专栏文章将讨论其它特性,比如 Tiger 中泛型类型的特定表现以及可能扩展为 Tiger 之外的泛型类型)。请通过单击文章顶部或底部的 讨论进入 论坛,与作者和其他读者分享您对本文的心得体会。

J2SE 1.5 - 代号为 Tiger - 计划在 2003 年年底发布。我一直都热衷于尽可能多地收集有关即将推出的新技术的预告信息,因此我将撰写一系列的文章,讨论可从 V1.5 中获得的新的和经过重组的特性,本文是第一篇。我特别想谈谈泛型类型并重点讲述在 Tiger 中为了支持它们而进行的更改和调整。

在许多方面,Tiger 肯定是迄今为止在 Java 编程方面(包括对源语言语法的重大扩展)所取得的最大进步。Tiger 中计划进行的最显著的变化是添加泛型类型,正如在 JSR-14 原型编译器中所预先展示的那样(您可以立即免费下载该编译器;请参阅 参考资料)。

让我们从介绍泛型类型是什么以及添加了什么特性来支持它们开始吧。

数据类型转换和错误

为理解泛型类型为何如此有用,我们要将注意力转向 Java 语言中最容易引发错误的因素之一 - 需要不断地将表达式向下类型转换(downcast)为比其静态类型更为具体的数据类型(请参阅 参考资料中的“The Double Descent bug pattern”,以了解进行数据类型转换时,可能会碰到的麻烦的某些方面)。

程序中的每个向下类型转换对于 ClassCastException 而言都是潜在的危险,应当尽量避免它们。但是在 Java 语言中它们通常是无法避免的,即便在设计优良的程序中也是如此。

在 Java 语言中进行向下类型转换最常见的原因在于,经常以专用的方式来使用类,这限制了方法调用所返回的参数可能的运行时类型。例如,假定往 Hashtable 中添加元素并从中检索元素。那么在给定的程序中,被用作键的元素类型和存储在散列表中的值类型,将不能是任意对象。通常,所有的键都是某一特定类型的实例。同样地,存储的值将共同具有比 Object 更具体的公共类型。

但是在目前现有的 Java 语言版本中,不可能将散列表的特定键和元素声明为比 Object 更具体的类型。在散列表上执行插入和检索操作的类型特征符告诉我们只能插入和删除任意对象。例如, putget 操作的说明如下所示:
清单 1. 插入/检索类型说明表明只能是任意对象

class Hashtable {
  Object put(Object key, Object value) {...}
  Object get(Object key) {...}
  ...
}

因此,当我们从类 Hashtable 的实例检索元素时,比如,即使我们知道在 Hashtable 中只放了 String ,而类型系统也只知道所检索的值是 Object 类型。在对检索到的值进行任何特定于 String 的操作之前,必须将它强制转换为 String ,即使是将检索到的元素添加到同一代码块中,也是如此!
清单 2. 将检索到的值强制转换成 String

import java.util.Hashtable;
class Test {
  public static void main(String[] args) {
    Hashtable h = new Hashtable();
    h.put(new Integer(0), "value");
    String s = (String)h.get(new Integer(0));
    System.out.println(s);
  }
}

请注意 main 方法主体部分的第三行中需要进行的数据类型转换。因为 Java 类型系统相当薄弱,因此代码会因象上面那样的数据类型转换而漏洞百出。这些数据类型转换不仅使 Java 代码变得更加拖沓冗长,而且它们还降低了静态类型检查的价值(因为每个数据类型转换都是一个选择忽略静态类型检查的伪指令)。我们该如何扩展该类型系统,从而不必回避它呢?




回页首

用泛型类型来解决问题!

要消除如上所述的数据类型转换,有一种普遍的方法,就是用 泛型类型来增大 Java 类型系统。可以将泛型类型看作是类型“函数”;它们通过类型变量进行参数化,这些类型变量可以根据上下文用各种类型参数进行 实例化

例如,与简单地定义类 Hashtable 不同,我们可以定义泛型类 Hashtable<Key, Value> ,其中 KeyValue 是类型参数。除了类名后跟着尖括号括起来的一系列类型参数声明之外,在 Tiger 中定义这样的泛型类的语法和用于定义普通类的语法很相似。例如,可以按照如下所示的那样定义自己的泛型 Hashtable 类:
清单 3. 定义泛型 Hashtable 类

class Hashtable<Key, Value> { ... }

然后可以引用这些类型参数,就像我们在类定义主体内引用普通类型那样,如下所示:
清单 4. 像引用普通类型那样引用类型参数

class Hashtable<Key, Value> {
  ...
  Value put(Key k, Value v) {...}
  Value get(Key k) {...}
}

类型参数的作用域就是相应类定义的主体部分(除了静态成员之外)(在下一篇文章中,我们将讨论为何 Tiger 实现中有这样的“怪习”,即必须对静态成员进行此项限制。请留意!)。

创建一个新的 Hashtable 实例时,必须传递类型参数以指定 KeyValue 的类型。传递类型参数的方式取决于我们打算如何使用 Hashtable 。在上面的示例中,我们真正想要做的是创建 Hashtable 实例,它只将 Integer 映射为 String 。可以用新的 Hashtable 类来完成这件事:
清单 5. 创建将 Integer 映射为 String 的实例

import java.util.Hashtable;
class Test {
  public static void main(String[] args) {
    Hashtable<Integer, String> h = new Hashtable<Integer, String>();
    h.put(new Integer(0), "value");
    ...
  }
}

现在不再需要数据类型转换了。请注意用来实例化泛型类 Hashtable 的语法。就像泛型类的类型参数用尖括号括起来那样,泛型类型应用程序的参数也是用尖括号括起来的。
清单 6. 除去不必要的数据类型转换

...
String s = h.get("key");
System.out.println(s);

当然,程序员若只是为了能使用泛型类型而必须重新定义所有的标准实用程序类(比如 HashtableList )的话,则可能会是一项浩大的工程。幸好,Tiger 为用户提供了所有 Java 集合类的泛型版本,因此我们不必自己动手来重新定义它们了。此外,这些类能与旧代码和新的泛型代码一起无缝工作(下个月,我们会说明如何做到这一点)。




回页首

Tiger 的基本类型限制

Tiger 中类型变量的限制之一就是,它们必须用引用类型进行实例化 - 基本类型不起作用。因此,在上面这个示例中,无法完成创建从 int 映射到 StringHashtable

这很遗憾,因为这意味着只要您想把基本类型用作泛型类型的参数,您就必须把它们组装为对象。另一方面,当前的这种情况是最糟的;您不能将 int 作为键传递给 Hashtable ,因为所有的键都必须是 Object 类型。

我们真正想看到的是,基本类型可以自动进行包装(boxing)和解包装(unboxing),类似于用 C# 所进行的操作(或者比后者更好)。遗憾的是,Tiger 不打算包括基本类型的自动包装(但是人们可以一直期待 Java 1.6 中出现该功能!)。




回页首

受限泛型

有时我们想限制可能出现的泛型类的类型实例化。在上面这个示例中,类 Hashtable 的类型参数可以用我们想用的任何类型参数进行实例化,但是对于其它某些类,我们或许想将可能的类型参数集限定为给定类型 范围内的子类型。

例如,我们可能想定义泛型 ScrollPane 类,它引用普通的带有滚动条功能的 Pane 。被包含的 Pane 的运行时类型通常会是类 Pane 的子类型,但是静态类型就只是 Pane

有时我们想用 getter 检索被包含的 Pane ,但是希望 getter 的返回类型尽可能具体些。我们可能想将类型参数 MyPane 添加到 ScrollPane 中,该类型参数可以用 Pane 的任何子类进行实例化。然后可以用这种形式的子句: extends Bound 来说明 MyPane 的声明,从而来设定 MyPane 的范围:
清单 7. 用 extends 子句来说明 MyPane 声明

class ScrollPane<MyPane extends Pane> { ... }

当然,我们可以完全不使用显式的范围,只要能确保没有用不适当的类型来实例化类型参数。

为什么要自找麻烦在类型参数上设定范围呢?这里有两个原因。首先,范围使我们增加了静态类型检查功能。有了静态类型检查,就能保证泛型类型的每次实例化都符合所设定的范围。

其次,因为我们知道类型参数的每次实例化都是这个范围之内的子类,所以可以放心地调用类型参数实例出现在这个范围之内的任何方法。如果没有对参数设定显式的范围,那么缺省情况下范围是 Object ,这意味着我们不能调用范围实例在 Object 中未曾出现的任何方法。




回页首

多态方法

除了用类型参数对类进行参数化之外,用类型参数对方法进行参数化往往也同样很有用。泛型 Java 编程用语中,用类型进行参数化的方法被称为 多态方法(Polymorphic method)

多态方法之所以有用,是因为有时候,在一些我们想执行的操作中,参数与返回值之间的类型相关性原本就是泛型的,但是这个泛型性质不依赖于任何类级的类型信息,而且对于各个方法调用都不相同。

例如,假定想将 factory 方法添加到 List 类中。这个静态方法只带一个参数,也将是 List 唯一的元素(直到添加了其它元素)。因为我们希望 List 成为其所包含的元素类型的泛型,所以希望静态 factory 方法带有类型变量 T 这一参数并返回 List<T> 的实例。

但是我们确实希望该类型变量 T 能在方法级别上进行声明,因为它会随每次单独的方法调用而发生改变(而且,正如我在下一篇文章中将讨论的那样,Tiger 设计的“怪习”规定静态成员不在类级类型参数的范畴之内)。Tiger 让我们通过将类型参数作为方法声明的前缀,从而在单独的方法级别上声明类型参数。例如,可以按照如下所示的那样为 factory 方法 make 添加前缀:
清单 8. 将类型参数作为前缀添加到方法声明

class Utilities {
   <T extends Object> public static List<T> make(T first) {
     return new List<T>(first);
   }
}

除了多态方法中所增加的灵活性之外,Tiger 中还增加了一个优点。Tiger 使用类型推断机制,根据参数类型来自动推断出多态方法的类型。这可以大大减少方法调用的繁琐和复杂性。例如,如果想调用 make 方法来构造包含 new Integer(0)List<Integer> 新实例,那么只需编写:
清单 9. 强制 make 构造新实例

Utilities.make(Integer(0))

然后会自动地从方法参数中推断出类型参数的实例化。




回页首

结束语

正如我们所见到的那样,在 Java 语言中添加泛型类型肯定会大大增强我们使用静态类型系统的能力。学习如何使用泛型类型相当简单,但是同样也需要避免一些缺陷。在接下来的文章中,我们将讨论如何充分使用将出现在 Tiger 中的泛型类型的特定表现,以及一些缺陷。我们还将研究对泛型 Java 类型工具的扩展,我们期盼这些工具可以出现在仍处于设计阶段的 Java 平台之中。

参考资料

  • 您可以参阅本文在 developerWorks 全球站点上的 英文原文.

  • 请参与有关本文的 论坛(您也可以单击文章顶部或底部的 讨论来访问该论坛)。

  • 通过下载 JSR-14 原型编译器(您必须是 Java Developer Connection 的注册成员)来进一步学习 Java 编程中的泛型。它包括了用扩展语言编写的原型编译器的源代码、包含了用于运行和自举编译器的类文件的 JAR 文件,以及包含了集合类存根的 JAR 文件。


  • Eric Allen 写了一本有关错误模式主题的新书: Bug Patterns in Java (Apress,2002),该书提出了一种诊断和调试计算机程序的方法论,这种方法论侧重于错误模式、极端编程方法和生成功能强大的、可测的且可扩展的软件的方法。


  • 请参阅“ Double Descent 错误模式”( developerWorks,2001 年 4 月),以了解在进行数据类型转换时,可能会碰到的麻烦的某些方面。


  • IntelliJ 的 IDEA 开发环境是值得一试的“好点子”,它包括了 J2EE 高速网络应用程序开发功能部件、一个功能强大的代码检查工具,以及一个用于第三方插件支持的开放式 API。


  • 并且别忘了尝试一下用于 J2SE 和 J2EE 开发的高性能代码分析引擎 - 来自 OmniCore 的 CodeGuide。它早已通过 JSR-14 原型编译器为 Java 代码中的泛型类型提供了 IDE 支持。


  • Martin Fowler 的 网站包含了许多有关有效重构的有用信息。


  • 研究“ 设计“可测试的”应用程序”( developerWorks,2001 年 9 月),以了解牢记测试来构建代码设计基础的七项原则。


  • 诊断 Java 代码专栏文章摘要 中,可以查阅 Eric Allen 专栏的 developerWorks 资源库 - 从错误模式到可测性再到设计策略。


  • 通过阅读 Java 社区过程(Java Community Process)的建议书: JSR-14来了解有关将泛型类型添加到 Java 代码中的讨论。


  • Keith Turner 在“ 编译时使用 Generic Java 捕获更多的错误”( developerWorks,2001 年 3 月)中提出了关于本主题的另一种观点。


  • 来自 IBM 研究部门(IBM Research)的论文“ Automatic Code Generation from Design Patterns”(PDF)描述了使设计模式实现自动化的工具的体系结构和实现。


  • Diagnosing Java code系列文章中的下面这两篇文章可以帮助您充实有关泛型类型和 Java 类型系统的知识:“ “杀手组合”― mixin、Jam 和单元测试”(2002 年 12 月)和“ 拥护静态类型的理由”(2002 年 6 月)。


  • developerWorksJava 技术专区 上查找其它大量的 Java 技术参考资料。
分享到:
评论

相关推荐

    诊断Java代码:轻松掌握Java泛型(一)

    ### 诊断Java代码:轻松掌握Java泛型(一) #### 数据类型转换和错误 在Java开发过程中,数据类型转换是一项常见的需求,尤其是在处理诸如`Object`等基础类型的对象时。当开发者希望对某个对象执行某种特定操作...

    java泛型和反射机制

    对java泛型以及反射机制进行原理和应用上的讲解,帮助初学者对这两个概念进行更轻松的掌握

    JAVA泛型与集合框架PPT课件PPT学习教案.pptx

    JAVA泛型与集合框架知识点总结 JAVA泛型是JDK1.5中引入的一种机制,主要目的是可以建立具有类型安全的集合...Java泛型和集合框架是Java语言的重要组成部分,掌握它们可以使得开发者更好地开发高效、可靠的应用程序。

    java中得泛型

    Java泛型的基本语法非常直观。例如,在定义一个泛型类时,我们可以在类名后面加上尖括号`&lt;&gt;`,并在尖括号内指定类型参数。这里的类型参数可以被视为一种占位符,表示将来使用这个类时将由具体类型替换的位置。例如:...

    VarJ:用于推断通配符的 Java 泛型重构工具

    Java泛型是Java编程语言中的一个关键特性,它允许程序员在定义类、接口和方法时指定...通过理解和应用VarJ,开发者能够更好地掌握Java泛型的使用,减少因类型推断不准确带来的问题,同时提升代码的可维护性和一致性。

    经典泛型dao层代码,非常好用简易

    ### 泛型DAO层在SSH框架中的应用与详解 #### 引言 在现代软件开发中,特别是基于Java的企业级应用开发中,...因此,对于那些追求高质量、高效率的Java企业级应用项目来说,掌握并应用泛型DAO层技术是非常有必要的。

    《轻松学Java》 PDF

    本教程《轻松学Java》旨在帮助初学者快速掌握Java编程的基础知识,并逐步提升至精通水平。 首先,Java编程的基础包括了解基本语法和数据类型。Java支持八种基本数据类型:整型(byte、short、int、long)、浮点型...

    Java轻松掌握(PDG)

    "Java轻松掌握(PDG)"很可能是一个Java学习资源,可能是电子书或者课程资料,其中包含了多个章节或部分,以PDG(可能代表Page Data Generator或某种特定的文件格式)文件呈现。 1. **基础概念**:在Java中,一切皆为...

    新手轻松自学JAVA教程.zip

    "新手轻松自学JAVA教程.zip" 是一个专门为初学者设计的资源包,旨在帮助那些对编程感兴趣但又缺乏经验的人快速掌握Java语言的基本概念和核心技能。 本教程可能包含了一系列的章节,从最基础的编程概念开始,逐步...

    headFirst java核心技术 java编程思想

    "Head First Java" 是一本以独特、直观的方式讲解Java的书籍,它通过生动的故事和丰富的视觉元素来吸引读者,使初学者能够更容易地理解和掌握Java的基础概念。书中涵盖了面向对象编程的基本原理,如类、对象、封装、...

    MapReduce,泛型,匿名内部类,Runnable和Callable

    MapReduce是一种分布式计算模型,由Google在2004年提出,主要用于处理和生成大规模数据集。它将复杂的并行计算任务分解为两个主要阶段:Map(映射)和...了解和掌握这些概念对于理解现代分布式系统和Java编程至关重要。

    Head First Java 中文高清版pdf

    这本书以其独特的教学方式,通过丰富的图像、幽默的插图和互动性的设计,帮助读者以轻松有趣的方式掌握Java编程的核心概念。 在Java的世界里,初学者通常会遇到诸如类、对象、继承、多态、接口等基础概念。《Head ...

    30天轻松掌握javaweb 课堂笔记完整版

    - 讲解了单元测试的概念以及如何在Java开发中使用Junit框架来提高代码质量和可靠性。 3. **Java新特性**: - 静态导入和自动装箱拆箱:介绍了Java 5中引入的静态导入功能以及自动装箱和拆箱的概念。 4. **增强...

    Java图解创意编程:从菜鸟到互联网大厂之路.pptx

    本书以图解的方式进行讲解,通过大量的实例和图表,使读者能够轻松理解和掌握Java编程的基础知识和技能。 知识点: 1. Java编程基础知识:变量、数据类型、控制流语句、函数、数组、链表、栈、队列、树等。 2. ...

    java学习,很轻松入门,快乐学习

    理解如何定义和使用类,以及如何通过继承和多态性来实现代码复用,是掌握Java的关键。同时,接口在Java中也扮演着重要角色,它提供了实现多继承的方式,并允许定义抽象方法,是实现设计模式的基础。 在Java中,异常...

    Head First Java(深入浅出 Java).第二版.中文完整高清

    这本书以其独特的教学方式,将复杂的编程概念以直观、生动的方式呈现,使得读者能够轻松理解和掌握Java语言的核心概念。 在深入浅出的讲解中,本书涵盖了Java编程的基础知识,包括: 1. **Java简介**:首先介绍...

    Head First Java 中文高清版

    本书采用轻松愉快的教学方式,寓教于乐,让读者能够在愉悦的阅读过程中掌握Java编程知识,从而达到学习的目的。这本书不但适合编程新手,也适合希望加深对Java理解的读者。 书中首先介绍了Java语言的基础知识,包括...

    java2轻松进阶

    《Java2轻松进阶》是一本专为Java初学者和有一定...通过学习和实践书中内容,读者将能够从初级程序员逐渐成长为熟练掌握Java2的开发者,具备解决实际问题的能力,为进一步深入学习Java EE或Android开发打下坚实基础。

Global site tag (gtag.js) - Google Analytics