成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

"一致性相等"的陷阱

開發 后端
方法equals()與Comparable接口中的compareTo()方法是Java中最基本的兩個方法之一,然而它們的定義卻圍繞著"與相等一致"這一有趣的概念。

關于Object類中的equals()方法與Comparable接口中的compareTo()方法之間有何種關聯,之前還真沒考慮過。通過java.net看到此文之后,收獲了一點兒新知識,希望大家也能如此。

方法equals()與Comparable接口中的compareTo()方法是Java中最基本的兩個方法之一,然而它們的定義卻圍繞著"與相等一致"這一有趣的概念。

equals()方法

Java中的equals()方法既明確,又模糊。Java清楚地定義了如何準確地檢驗一個equals()方法是可用的。一個恰當的equals()方法必須是自反的,對稱的,可傳遞的,一致的,并能處理null引用。

然而equals()方法又是不清晰的。Javadoc說到,該方法指定了其它對象是"等于"這個對象的。注意,"等于"是放在引號中的。此處的關鍵就是,它沒有定義如何去判定這種相等性。

·對象的一致性(==)默認是繼承自Object類

·對象的整體可觀測的狀態,例如,若兩個對象是相等的,那么在應用的其它部分可以用一個對象去替代另一個對象。

·對象信息中的某些部分,如ID,使得檢驗對象相等性在邏輯上是有意義的。

compareTo()方法

Comparable接口定義了可比較性的概念。Javadoc指出compareTo()方法"強制設定了每個實現了該接口的類的對象的全部順序"。

實現了Comparable接口的類有一個天然的排序,這可便于存儲,也能在不使用單獨的Comparator的情況下,用于像TreeSet和TreeMap這樣的集合對象。

該接口的定義明晰,它要求其實現必須確保對稱性與傳遞性,就像equals()方法那樣。

一致性/非一致性相等

Comparable接口有如下描述

類C的天然排序意味著要與equals()方法保持一致,只有當且僅當e1.compareTo(e2) == 0與e1.equals(e2)有相同的布爾值。

基本上,這就要求由compareTo()定義的相等性與equals()方法定義的相等性具有相同的概念(除去有null的情況)。乍一看,該要求很簡單,但實際上它有其復雜性,后面將會討論到。

當考慮到操作符重載時,這種定義就特別有用。若我們假設有一種類Java語言,在這種語言中,==并不表示對象的同一性,而是通過方法去進行比較,大于/小于操作符也是如此,問題是調什么樣的方法。在類Java語言中大于/小于天然地就要基于compareTo()方法,而==則要調用equals()方法。

  1. // our new Java-like language  
  2. if (a < b) return "Less";      // translation ignoring nulls: if (a.compareTo(b) < 0)  
  3. if (a > b) return "Greater";   // translation ignoring nulls: if (a.compareTo(b) > 0)  
  4. if (a == b) return "Equal";    // translation ignoring nulls: if (a.equals(b))  
  5. throw new Exception("Impossible assuming no nulls?"); 

但如果compareTo()方法不是"一致性相等",那么上述代碼將會拋出異常,因為當a.equals(b)為false時,a.compareTo(b)會返回0。

在集合,如TreeMap,中還會發生其它問題:

  1. // Foo class is "inconsistent with equals"  
  2. assert foo1.equals(foo2) == false;  
  3. assert foo1.compareTo(foo2) == 0;  
  4.    
  5. TreeMap<Foo, String> map =   
  6. map.put(foo1, "a");  
  7. map.put(foo2, "b"); 

當使用equals()方法時,這兩個對象不相等,但使用compareTo()時,它們卻相等。在這種情況下,該Map的元素個數將為1,而非0。

由于這些"一致性相等"的問題,Javadoc說道"強烈建議(盡管并不要求)天然排序規則要與equals()方法保持一致"。

JDK中的許多類為了符合"一致性相等"這一規范而實現了Comparable接口。這些類包括Byte,Short,Integer,Long,Character和String。

還有些更有趣的類:

BigDecimal--肯定是"非一致性相等",比如4.00與4.0不一致,但進行比較時,認為它們是一樣的。

Double/Float--該類顯式地提供了排序規則,并為正零和負零,以及NaN都提供了相等性檢查,以確保它的compareTo()方法符合"一致性相等"。

CharSet--該類基于ID或名稱。equals()方法對待字段串是大小寫敏感的,但compareTo()方法卻不這樣。雖然名稱一般會符合某種標準,但這是一種值得懷疑的"一致性"。

*Buffer(nio)--該簇類的比較基于緩沖存放的內容,在我的測試中equals()和compareTo()是"一致的"。

Rdn(ldap)--該類的比較基于狀態的標準化格式,因此也是"一致性相等"。

ObjectStreamField(序列化)--該類的比較基于名稱,但會首先對基本數據類型進行排序。因為沒有覆蓋equals()方法,所以是"非一致性相等"。

 ...

注意:對于大多數的例子,我都不得不查看其源代碼或編寫測試程序以確定該類是不是符合"一致性相等"。這兒有一個不錯地清理Javadoc和檢驗UUID equals()方法的Adopt-a-JDK任務。

JSR-310

一直看到許多關于BigDecimal的問題,已有計劃將JSR-310中的類改造成"一致性相等",最近的一些帖子顯示這將造成多么大的爭議。

基本上,為某些類定義equals()和compareTo()看起來很容易。LocalDate表示某單一日歷系統中的某個日期,所以它有一個顯而易見的排序算法和相等規則。LocalTime則表示某個時刻,所以它也有一個明顯的排序算法和相等規則。Instant表示時間線上的某個時刻,那么它的排序與相等也是顯見的。

但在其它的情況下,這就不是那么顯而易見了。考慮這樣一個類OffsetDateTime:

  1. dt1 = OffsetDateTime.parse("2012-11-05T06:00+01:00");  
  2. dt2 = OffsetDateTime.parse("2012-11-05T07:00+02:00"); 

這樣的兩個日期-時刻對象代表時間線上一個相同的時刻點,但它們有不同的本地時,而且相對的UTC/格林威治時間的偏移量也不相同。

那么就有一個問題要留給讀者們...你更傾向于如下哪種觀點...

1. dt1不等于dt2,compareTo()分別比較本地時與偏移量,使用"一致性相等"(使用獨立的Comparator基于時刻對其進行排序)。

2. dt1不等于dt2,compareTo()基于時間線的上時刻點,使用"非一致性相等"。

3. dt1等于dt2,compareTo()基于時間線的上時刻點,使用"一致性相等"。

4. dt1等于dt2,且不實現Comparable接口。

5. dt1不等于dt2,且不實現Comparable接口。

我個人更傾向于讓dt1.equals(dt2)返回true這種方案,但我仍持開放態度。

順便地,也可以將這個問題提給BigDecimal,如果你能修改這個類,使其符合"一致性相等",你會修改它的equals()方法,還是compareTo()方法?

原文鏈接:http://www.blogjava.net/jiangshachina/archive/2012/12/06/392569.html

責任編輯:張偉 來源: blogjava
相關推薦

2017-07-25 14:38:56

數據庫一致性非鎖定讀一致性鎖定讀

2022-12-14 08:23:30

2021-02-05 08:00:48

哈希算法?機器

2021-02-02 12:40:50

哈希算法數據

2021-07-26 06:33:42

CRDT數據CAP

2021-06-22 10:22:08

業務IT一致性首席信息官

2020-11-24 09:03:41

一致性MySQLMVCC

2020-05-12 10:43:22

Redis緩存數據庫

2024-04-11 13:45:14

Redis數據庫緩存

2021-06-30 21:13:49

CPUCache數據

2022-10-19 12:22:53

并發扣款一致性

2022-03-22 09:54:22

Hash算法

2019-08-30 12:46:10

并發扣款查詢SQL

2020-04-01 15:50:17

TiDBMySQL數據庫

2021-02-04 06:30:26

Python編程語言

2021-12-26 00:48:05

一致性視圖數據庫

2025-03-27 08:20:54

2020-08-05 08:46:10

NFS網絡文件系統

2021-05-19 21:50:46

Hash算法測試

2023-06-25 09:44:00

一致性哈希數據庫
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: av中文字幕在线 | 伊人热久久 | www免费视频 | 久久久亚洲成人 | 久久中文字幕视频 | 一区二区三区亚洲 | 久久r精品 | 在线免费观看黄a | 色黄网站 | 午夜免费视频 | 青青久久| 99久久成人| 一区二区三区日韩精品 | 91精品一区| 国产伦精品一区二区三区视频金莲 | 精品一二区 | 亚洲综合大片69999 | 日韩成人在线免费观看 | 久久这里只有精品首页 | av影片在线| 日本高清不卡视频 | 国产精品久久av | 在线观看视频91 | 亚洲国内精品 | 精品视频一区二区在线观看 | 日日摸夜夜爽人人添av | 午夜精品久久久久久久久久久久 | 久久精品国产免费一区二区三区 | 久久精品一区二区三区四区 | 国产精品夜夜春夜夜爽久久电影 | 婷婷成人在线 | 在线观看成人小视频 | 日韩成人影院在线观看 | 国产视频一区二区三区四区五区 | 欧美色偷拍 | 免费成人国产 | 国产成人艳妇aa视频在线 | 日本电影韩国电影免费观看 | 色婷婷综合久久久中文字幕 | 中文字幕一区在线观看视频 | 欧洲一区二区视频 |