文档

Java™ 教程-Java Tutorials 中文版
原始类型
Trail: Learning the Java Language
Lesson: Generics (Updated)
Section: Generic Types

原始类型

raw type (原始类型) 是没有任何类型实参的泛型类或接口的名称。例如,给定泛型 Box 类:

public class Box<T> {
    public void set(T t) { /* ... */ }
    // ...
}

要创建 Box<T> 的参数化类型,需要为形式类型形参 T 提供实际类型实参:

Box<Integer> intBox = new Box<>();

如果省略实际类型实参,则创建 Box<T> 的原始类型:

Box rawBox = new Box();

因此,Box 是泛型类型 Box<T> 的原始类型。但是,非泛型类或接口类型 不是 原始类型。

原始类型显示在遗留代码中,因为许多 API 类(例如 Collections 类)在 JDK 5.0 之前不是泛型的。使用原始类型时,基本上可以获得 pre-generics 行为 - Box 为你提供 Object。为了向后兼容,允许将参数化类型赋值给其原始类型:

Box<String> stringBox = new Box<>();
Box rawBox = stringBox;               // OK

但是,如果将原始类型分配给参数化类型,则会收到警告:

Box rawBox = new Box();           // rawBox is a raw type of Box<T>
Box<Integer> intBox = rawBox;     // warning: unchecked conversion

如果使用原始类型调用相应泛型类型中定义的泛型方法,也会收到警告:

Box<String> stringBox = new Box<>();
Box rawBox = stringBox;
rawBox.set(8);  // warning: unchecked invocation to set(T)

警告显示原始类型绕过了泛型类型检查,将不安全代码的捕获推迟到运行时。因此,你应该避免使用原始类型。

Type Erasure 部分提供了有关 Java 编译器如何使用原始类型的更多信息。

未经检查的错误信息

如前所述,在将遗留代码与泛型代码混合时,你可能会遇到类似于以下内容的警告消息:

Note: Example.java uses unchecked or unsafe operations.
Note: Recompile with -Xlint:unchecked for details.

使用在原始类型上运行的旧 API 时会发生这种情况,如以下示例所示:

public class WarningDemo {
    public static void main(String[] args){
        Box<Integer> bi;
        bi = createBox();
    }

    static Box createBox(){
        return new Box();
    }
}

术语 "unchecked (未经检查的)" 表示编译器没有足够的类型信息来执行确保类型安全所必需的所有类型检查。默认情况下,"unchecked" 警告被禁用,尽管编译器会提示。要查看所有 "未经检查的" 警告,请使用 -Xlint:unchecked 重新编译。

使用 -Xlint:unchecked 重新编译前一个示例会显示以下附加信息:

WarningDemo.java:4: warning: [unchecked] unchecked conversion
found   : Box
required: Box<java.lang.Integer>
        bi = createBox();
                      ^
1 warning

要完全禁用未经检查的警告,请使用 -Xlint:-unchecked 标志。@SuppressWarnings("unchecked") 注解会抑制未经检查的警告。如果你不熟悉 @SuppressWarnings 语法,请参阅 Annotations


Previous page: Generic Types
Next page: Generic Methods