Java基础知识18-泛型

2017-10-31  本文已影响25人  我相信你爱过gg

什么是泛型

就本质而言, 术语"泛型"的意思是参数化类型. 参数化类型很重要, 因为使用该特性创建的类, 接口以及方法, 可以作为参数指定所操作数据的类型.

一个简单的泛型示例

class Gen<T> {
    T ob;

    Gen(T o) {
        this.ob = o;
    }

    T getOb() {
        return this.ob;
    }

    void showType() {
        System.out.println("Type of T is " + this.ob.getClass().getName());
    }
}

public class GenDemo {
    public static void main(String[] args) {
        Gen<Integer> iOb = new Gen<Integer>(88);
        iOb.showType();

        int v = iOb.getOb();
        System.out.println("value: " + v);

        System.out.println();

        Gen<String> strOb = new Gen<>("Generics Test");
        strOb.showType();

        String str = strOb.getOb();
        System.out.println("value: " + str );
    }
}

下面详细分析该程序.
首先, 注意下面这行代码声明泛型类 Gen 的方式:

class Gen<T> {

其中, T 是类型参数的名称. 这个名称是实际类型的占位符, 当创建对象时, 将实际类型传递给 Gen. 因此在 Gen 中, 只需要类型参数, 就是用 T.

注意 T 被包含在<>中, 只要是声明类型参数, 就需要在尖括号中指定, 因为 Gen 使用类型参数, 所以 Gen 是泛型类, 也成为参数化类型.

接下来使用 T 声明 ob, 如下所示:

T ob;

前面解释过, T 是将在创建 Gen 对象时指定的实际类型的占位符. 因此, ob 是传递给 T 的那种实际类型的对象. 例如, 如果将String 传递给 T , ob 将是String 类型.

现在分析 Gen 的构造函数:

    Gen(T o) {
        this.ob = o;
    }

注意参数 o 的类型是 T, 这意味着 o 的实际类型取决于创建 Gen 对象时传递给 T 的类型. 此外, 因为参数 o 和成员变量 ob 的类型都是T, 所以在创建 Gen 对象时, 它们将具有相同的类型.

还可以使用类型参数 T 指定方法的返回类型, 就像 getOb() 方法那样, 如下所示:

    T getOb() {
        return this.ob;
    }

因为 ob 也是T类型, 所以 ob 的类型和 getOb() 方法指定的返回类型是兼容的.

showType() 方法通过对 Class 对象调用 getName() 方法显式 T 的类型, 而这个 Class 对象是通过对 ob 调用 getClass() 方法返回的.

GenDemo 类演示了泛型化的 Gen 类. 它首先创建整形版本的 Gen类, 如下所示:

Gen<Integer> iOb

请仔细分析这个声明. 首先, 注意类型 Integer 是在 Gen 后面的尖括号中指定. 在此, Integer 是传递给 Gen 的类型参数. 对T 的所有引用都被转化为对 Integer 的引用. 因此对于这个声明, ob 是 Integer 类型, 并且 getob() 方法的返回类型是 Integer.

该代码将一个引用赋给 iOb:

iOb = new Gen<Integer>(88);

注意在调用 Gen 构造函数时, 仍然指定了类型参数 Integer. 这是必须的, 因为将其赋值的对象类型是 Gen<Integer>. 因此, new 返回的引用必须是 Gen<Integer> 类型.如果不是的话, 就会产生编译错误.

接下来, 程序使用下面这行代码获取 ob 的值:

int v = iOb.getOb();

因为 getob() 方法的返回类型是 T, 当声明 iOb 时 T 已被替换为 Integer 类型, 所以 getob() 方法的返回类型也是 Integer, 当返回值赋值给 v 时会自动拆箱为 int 类型.

接下来, GemDemo 声明了 Gem<String> 类型的对象:

Gen<String> strOb = new Gen<>("Generics Test");

因为类型参数是 String, 所以使用 String 替换 Gen 中的 T,这会创建Gen的String版本, 就像程序中剩余代码所演示的那样.

泛型使用引用类型

当声明泛型类的实例时, 传递过来的参数必须是引用类型. 不能使用基本类型, 如果 int 或 char. 例如, 对 Gen, 可以将任何类型传递给 T, 但是不能将基本类型传递给类型参数T ,所以, 一下声明是非法的.

Gen<int> intOb = new Gen<int>(53);

带两个类型参数的泛型类

class TwoGen<T, V> {
    T ob1;
    V ob2;

    TwoGen (T o1, V o2) {
        this.ob1 = o1;
        this.ob2 = o2;
    }

    void showTypes() {
        System.out.println("Type of T is " + ob1.getClass().getName());
        System.out.println("Type of V is " + ob2.getClass().getName());
    }

    T getOb1() {
        return ob1;
    }

    V getOb2() {
        return ob2;
    }

}


public class SimpGen {
    public static void main(String[] args) {

        TwoGen<Integer, String> tgObj =
                new TwoGen<>(88, "Generics");

        tgObj.showTypes();

        int v = tgObj.getOb1();
        System.out.println("value: " + v);

        String str = tgObj.getOb2();
        System.out.println("value: " + str);
    }
}

注意 TwoGen 的声明方式:

class TwoGen<T, V> {

在此指定了两个类型参数: T 和 V, 使用逗号将他们隔开. 创建对象时必须为 TwoGen 传递两个类型参数, 如下所示:

TwoGen<Integer, String> tgObj =
                new TwoGen<>(88, "Generics");

在此, Integer 替换 T, String 替换 V.
在这个例子中, 尽管两个类型参数是不同的, 但是可以将两个类型参数设置为相同的类型. 例如, 下面这行代码是合法的:

TwoGen<String, String> x=
                new TwoGen<>("A", "B");

在此, T 和 V 都是 String 类型. 当然, 如果类型参数总是相同的, 就不必使用两个类型参数了.

上一篇下一篇

猜你喜欢

热点阅读