亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定

程序員的職業素養真是完全不能相信的東西

標簽:
Java

事情要从一次面试说起,面试官问了这么一个问题,在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却不是List子类,一个能比较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的代码契约一样,如果编译器在语法层面完全无法提供检查,只能靠程序员的职业素养来产生良好的代码,真是令人太不开心了。

點擊查看更多內容
TA 點贊

若覺得本文不錯,就分享一下吧!

評論

作者其他優質文章

正在加載中
  • 推薦
  • 評論
  • 收藏
  • 共同學習,寫下你的評論
感謝您的支持,我會繼續努力的~
掃碼打賞,你說多少就多少
贊賞金額會直接到老師賬戶
支付方式
打開微信掃一掃,即可進行掃碼打賞哦
今天注冊有機會得

100積分直接送

付費專欄免費學

大額優惠券免費領

立即參與 放棄機會
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號

舉報

0/150
提交
取消