本月的 诊断 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
更具体的类型。在散列表上执行插入和检索操作的类型特征符告诉我们只能插入和删除任意对象。例如, put
和 get
操作的说明如下所示:
清单 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);
}
}
|
用泛型类型来解决问题!
要消除如上所述的数据类型转换,有一种普遍的方法,就是用 泛型类型来增大 Java 类型系统。可以将泛型类型看作是类型“函数”;它们通过类型变量进行参数化,这些类型变量可以根据上下文用各种类型参数进行 实例化。
例如,与简单地定义类 Hashtable
不同,我们可以定义泛型类 Hashtable<Key, Value>
,其中 Key
和 Value
是类型参数。除了类名后跟着尖括号括起来的一系列类型参数声明之外,在 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
实例时,必须传递类型参数以指定 Key
和 Value
的类型。传递类型参数的方式取决于我们打算如何使用 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);
|
当然,程序员若只是为了能使用泛型类型而必须重新定义所有的标准实用程序类(比如 Hashtable
和 List
)的话,则可能会是一项浩大的工程。幸好,Tiger 为用户提供了所有 Java 集合类的泛型版本,因此我们不必自己动手来重新定义它们了。此外,这些类能与旧代码和新的泛型代码一起无缝工作(下个月,我们会说明如何做到这一点)。
Tiger 的基本类型限制
Tiger 中类型变量的限制之一就是,它们必须用引用类型进行实例化 - 基本类型不起作用。因此,在上面这个示例中,无法完成创建从 int
映射到 String
的 Hashtable
。
这很遗憾,因为这意味着只要您想把基本类型用作泛型类型的参数,您就必须把它们组装为对象。另一方面,当前的这种情况是最糟的;您不能将 int
作为键传递给 Hashtable
,因为所有的键都必须是 Object
类型。
我们真正想看到的是,基本类型可以自动进行包装(boxing)和解包装(unboxing),类似于用 C# 所进行的操作(或者比后者更好)。遗憾的是,Tiger 不打算包括基本类型的自动包装(但是人们可以一直期待 Java 1.6 中出现该功能!)。
受限泛型
有时我们想限制可能出现的泛型类的类型实例化。在上面这个示例中,类 Hashtable
的类型参数可以用我们想用的任何类型参数进行实例化,但是对于其它某些类,我们或许想将可能的类型参数集限定为给定类型 范围内的子类型。
例如,我们可能想定义泛型 ScrollPane
类,它引用普通的带有滚动条功能的 Pane
。被包含的 Pane
的运行时类型通常会是类 Pane
的子类型,但是静态类型就只是 Pane
。
有时我们想用 getter 检索被包含的 Pane
,但是希望 getter 的返回类型尽可能具体些。我们可能想将类型参数 MyPane
添加到 ScrollPane
中,该类型参数可以用 Pane
的任何子类进行实例化。然后可以用这种形式的子句: extendsBound
来说明 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代码:轻松掌握Java泛型(一) #### 数据类型转换和错误 在Java开发过程中,数据类型转换是一项常见的需求,尤其是在处理诸如`Object`等基础类型的对象时。当开发者希望对某个对象执行某种特定操作...
对java泛型以及反射机制进行原理和应用上的讲解,帮助初学者对这两个概念进行更轻松的掌握
JAVA泛型与集合框架知识点总结 JAVA泛型是JDK1.5中引入的一种机制,主要目的是可以建立具有类型安全的集合...Java泛型和集合框架是Java语言的重要组成部分,掌握它们可以使得开发者更好地开发高效、可靠的应用程序。
Java泛型的基本语法非常直观。例如,在定义一个泛型类时,我们可以在类名后面加上尖括号`<>`,并在尖括号内指定类型参数。这里的类型参数可以被视为一种占位符,表示将来使用这个类时将由具体类型替换的位置。例如:...
Java泛型是Java编程语言中的一个关键特性,它允许程序员在定义类、接口和方法时指定...通过理解和应用VarJ,开发者能够更好地掌握Java泛型的使用,减少因类型推断不准确带来的问题,同时提升代码的可维护性和一致性。
### 泛型DAO层在SSH框架中的应用与详解 #### 引言 在现代软件开发中,特别是基于Java的企业级应用开发中,...因此,对于那些追求高质量、高效率的Java企业级应用项目来说,掌握并应用泛型DAO层技术是非常有必要的。
本教程《轻松学Java》旨在帮助初学者快速掌握Java编程的基础知识,并逐步提升至精通水平。 首先,Java编程的基础包括了解基本语法和数据类型。Java支持八种基本数据类型:整型(byte、short、int、long)、浮点型...
"Java轻松掌握(PDG)"很可能是一个Java学习资源,可能是电子书或者课程资料,其中包含了多个章节或部分,以PDG(可能代表Page Data Generator或某种特定的文件格式)文件呈现。 1. **基础概念**:在Java中,一切皆为...
"新手轻松自学JAVA教程.zip" 是一个专门为初学者设计的资源包,旨在帮助那些对编程感兴趣但又缺乏经验的人快速掌握Java语言的基本概念和核心技能。 本教程可能包含了一系列的章节,从最基础的编程概念开始,逐步...
本文将深入探讨如何通过正确的资源和高效的学习方法,在较短的时间内轻松掌握Java,并以此为契机获得理想的高薪工作。 #### Java基础知识与核心概念 Java是一种面向对象的编程语言,拥有自动内存管理机制,这使得...
"Head First Java" 是一本以独特、直观的方式讲解Java的书籍,它通过生动的故事和丰富的视觉元素来吸引读者,使初学者能够更容易地理解和掌握Java的基础概念。书中涵盖了面向对象编程的基本原理,如类、对象、封装、...
MapReduce是一种分布式计算模型,由Google在2004年提出,主要用于处理和生成大规模数据集。它将复杂的并行计算任务分解为两个主要阶段:Map(映射)和...了解和掌握这些概念对于理解现代分布式系统和Java编程至关重要。
这本书以其独特的教学方式,通过丰富的图像、幽默的插图和互动性的设计,帮助读者以轻松有趣的方式掌握Java编程的核心概念。 在Java的世界里,初学者通常会遇到诸如类、对象、继承、多态、接口等基础概念。《Head ...
- 讲解了单元测试的概念以及如何在Java开发中使用Junit框架来提高代码质量和可靠性。 3. **Java新特性**: - 静态导入和自动装箱拆箱:介绍了Java 5中引入的静态导入功能以及自动装箱和拆箱的概念。 4. **增强...
本书以图解的方式进行讲解,通过大量的实例和图表,使读者能够轻松理解和掌握Java编程的基础知识和技能。 知识点: 1. Java编程基础知识:变量、数据类型、控制流语句、函数、数组、链表、栈、队列、树等。 2. ...
理解如何定义和使用类,以及如何通过继承和多态性来实现代码复用,是掌握Java的关键。同时,接口在Java中也扮演着重要角色,它提供了实现多继承的方式,并允许定义抽象方法,是实现设计模式的基础。 在Java中,异常...
这本书以其独特的教学方式,将复杂的编程概念以直观、生动的方式呈现,使得读者能够轻松理解和掌握Java语言的核心概念。 在深入浅出的讲解中,本书涵盖了Java编程的基础知识,包括: 1. **Java简介**:首先介绍...
本书采用轻松愉快的教学方式,寓教于乐,让读者能够在愉悦的阅读过程中掌握Java编程知识,从而达到学习的目的。这本书不但适合编程新手,也适合希望加深对Java理解的读者。 书中首先介绍了Java语言的基础知识,包括...
《Java2轻松进阶》是一本专为Java初学者和有一定...通过学习和实践书中内容,读者将能够从初级程序员逐渐成长为熟练掌握Java2的开发者,具备解决实际问题的能力,为进一步深入学习Java EE或Android开发打下坚实基础。