美文网首页互联网科技程序猿阵线联盟-汇总各类技术干货
程序员的职业素养真是完全不能相信的东西——浅谈Java泛型编程中

程序员的职业素养真是完全不能相信的东西——浅谈Java泛型编程中

作者: 大神带我来搬砖 | 来源:发表于2018-07-07 22:09 被阅读807次

事情要从一次面试说起,面试官问了这么一个问题,在JDK下面这个方法中:

public static <T extends Comparable<? super T>> void sort(List<T> list) 

这里面<T extends Comparable<? super T>>有什么用?

美好的愿景

很明显,从语义上来说,sort方法要对一个List排序,这个List中的元素类型为T,sort方法要求这个T类型必须是可比较的。这个可比较的含义是说,任意两个T类型的对象,可以通过compareTo方法来确定大小。理想情况下,一个T类型实现了Comparable接口的话,那么对于任意两个属于T类型的对象x和y,都可以通过compareTo方法来确定大小。因此,方法写成如下形式即可。

public static <T extends Comparable> void sort(List<T> list) 

Comparable接口的无奈

可是现实世界里,并不是T类型实现了Comparable接口就可以和T类型比较,而是T类型实现Comparable<E>之后,可以和E比较。尽管T实现Comparable<E>毫无道理可言,但是在语法上是正确的。
所以对于下面这个类,只能说代码写得烂,作者职业素养低,而编译器却无能为力,令人扼腕叹息。

public class Bar implements Comparable<String>

Java泛型系统出来背一次锅

为了将这样的类拒之门外,sort方法写成下面这样岂不是很好?sort需要的T类型,是一个能和T类型比较的类型,真是天衣无缝啊。

public static <T extends Comparable<T>> void sort(List<T> list) 

但是有这样一种情况,假设有个类型S,S实现了Comparable<S>,然后T继承了S,那么T就具备了和S比较的能力。但是这时T能通过编译么?答案是不能,因为在编译器看来,T只具备和S比较的能力,不具备和T比较的能力。尽管T是S的子类,但是编译器不认为如果一个类型具备了和S比较的能力,就具备了和T比较的能力。听上去感觉有点不合理,就好比经常有猎头问我:“好了,我现在知道你会用hadoop、spark这些来搞大数据了,那么请问你会Java么?”

这个就是Java泛型系统的一个特性——不具备协变、逆变的能力,所以尽管Integer是Number的子类,List<Integer>却不是List<Number>子类,一个能比较Number的比较器也不能被当做一个Integer的比较器。

所以最终用<T extends Comparable<? super T>>对T进行了约束,这样不管T的Comparable是自己实现的也好,还是继承的也好,都可以海纳百川有容乃大的被sort方法接受了。

美中不足

但是<T extends Comparable<? super T>>约束能力还是不够强,因为它在语义上表示的是“T能够和super T的任何类型比较”,所以下面这个类还是可以编译通过的,除了你的同事会对你的编程水平和职业素养产生质疑。

public class Bar extends RuntimeException implements Comparable<Exception>

就像equals和hashcode的代码契约一样,如果编译器在语法层面完全无法提供检查,只能靠程序员的职业素养来产生良好的代码,真是令人太不开心了。

相关文章

  • 程序员的职业素养真是完全不能相信的东西——浅谈Java泛型编程中

    事情要从一次面试说起,面试官问了这么一个问题,在JDK下面这个方法中: 这里面

  • Java编程思想---泛型(3)

    Java编程思想---泛型(3) 泛型带来的各类问题 任何基本类型都不能作为类型参数 Java泛型的限制之一是不能...

  • java 泛型解析

    Java 泛型 1、泛型的精髓是什么 2、泛型方法如何使用 概述: 泛型在java中具有重要地位,在面向对象编程模...

  • Java 中的泛型

    1、泛型的概述: 1.1 泛型的由来 根据《Java编程思想》中的描述,泛型出现的动机:有很多原因促成了泛型的出现...

  • Java泛型使用示例整理

    Java泛型使用示例整理 目标 Java泛型编程是JDK1.5版本后引入的。泛型让编程人员能够使用类型抽象,通常用...

  • kotlin中的reified关键字

    说kotlin中这个关键字之前先简单说下Java中的泛型,我们在编程中,出于复用和高效的目的,经常使用泛型。泛型是...

  • java中泛型(上)

    本文参考了java核心卷1中泛型程序设计该章节和java编程思想第15章。 泛型即适用于许多许多的类型。泛型最初的...

  • java泛型中类型擦除的一些思考

    java泛型 java泛型介绍 java泛型的参数只可以代表类,不能代表个别对象。由于java泛型的类型参数之实际...

  • Java语言中的Type

    Java中的Type 简介 泛型引入之前,Java中的类型系统很简单: 完全取决于Class,泛型引入后,变为以j...

  • Java中的泛型/范型

    维基百科中关于Java泛型的描述 Java 泛型的参数只可以代表类,不能代表个别对象。由于Java泛型的类型参数之...

网友评论

本文标题:程序员的职业素养真是完全不能相信的东西——浅谈Java泛型编程中

本文链接:https://www.haomeiwen.com/subject/jalguftx.html