java - 理解java中的有界泛型。重点是什么?

标签 java generics parameterized-types bounded-types

我试图理解有界类型,但不太了解它们的重点。

有一个example提供此用例的有界泛型:

public class NaturalNumber<T extends Integer> {

    private T n;

    public NaturalNumber(T n)  { this.n = n; }

    public boolean isEven() {
        return n.intValue() % 2 == 0;
    }

    // ...
}

如果您要限制可以是参数化类型的类,为什么不一起忘记参数化并具有:
public class NaturalNumber {

    private Integer n;

    public NaturalNumber(Integer n)  { this.n = n; }

    public boolean isEven() {
        return n.intValue() % 2 == 0;
    }

    // ...
}

然后任何扩展/实现的类 Integer可以与这个类一起使用。

还有一个附带问题:T 怎么样?延伸 Integer在第一个例子中,当 Java Integer类(class)是 final ?

最佳答案

How is T extending Integer in the first example when the Java Integer class is final?


T只能是 Integer ,所以这里的“扩展”纯粹是象征性的。 (我从旁注开始,因为事实上,这是一个泛型无用的例子。我真的不知道为什么教程认为这是一个信息丰富的演示。它不是。)

假设 T extends Number :
class Example<T extends Number> {
    private T num;

    void setNum(T num) { this.num = num; }
    T    getNum()      { return num;     }
}

所以泛型的要点是,你可以这样做:
Example<Integer> e = new Example<>();
e.setNum( Integer.valueOf(10) );
// returning num as Integer
Integer i = e.getNum();
// and this won't compile
e.setNum( Double.valueOf(10.0) );

泛型是 parametric polymorphism 的一种形式,本质上它让我们可以重用代码,并就所涉及的类型具有普遍性。

那么边界有什么意义呢?

这里的边界意味着 T必须是 NumberNumber 的子类,所以我们可以调用 Number 的方法在 T 的实例上. Number不幸的是,它本身通常是一个无用的基类(因为精度问题),但它可能让我们做一些有趣的事情,例如:
class Example<T extends Number> extends Number {
//                              ^^^^^^^^^^^^^^
    ...
    @Override
    public int intValue() {
        return num.intValue();
    }
    // and so on
}

例如,更常见的是查找 T extends Comparable<T>这让我们可以用 T 做一些更有意义的事情.我们可能有这样的事情:
// T must be a subclass of Number
// AND implement Comparable
Example<T extends Number & Comparable<T>>
        implements Comparable<Example<T>> {
    ...
    @Override
    public int compareTo(Example<T> that) {
        return this.num.compareTo(that.num);
    }
}

现在我们的 Example类具有自然顺序。我们可以对它进行排序,即使我们不知道是什么 T实际上是在类体内。

如果我们结合这些概念,那就:
  • 泛型允许“外部世界”指定实际类型和
  • 边界允许“内部世界”使用共性,

  • 我们可以构建结构,例如:
    static <T extends Comparable<T>> T min(T a, T b) {
        return (a.compareTo(b) < 0) ? a : b;
    }
    
    {
        // returns "x"
        String s = min("x", "z");
        // returns -1
        Integer i = min(1, -1);
    }
    

    关于java - 理解java中的有界泛型。重点是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30292959/

    相关文章:

    java - 使用通配符泛型编译

    参数化类的java流返回数组

    java - 模拟构建类 AWS Logs 客户端构建器

    java - 更新 Android 应用中的原始资源

    c# - 如何在 C# 中编写实现给定接口(interface)的通用容器类?

    java - 什么是原始类型,为什么我们不应该使用它呢?

    java - 将不同的参数化类型存储在 Java 中的同一个泛型数组列表中

    java - Android 的 ListView 。 AdapterView.OnItemClickListener 中的参数化类型

    java - 保护 JAAS 配置文件的安全

    java - Java socket 多线程安全吗?