Java中替代equals,compareTo和toString的方法
我們都曾在POJO中重寫過(guò)equals(),compareTo()和toString()方法。但是另有其他能做到職責(zé)分離的更好的方法并帶來(lái)更簡(jiǎn)潔的代碼。閱讀這篇文章來(lái)一探究竟吧!
更簡(jiǎn)明的職責(zé)——擺脫equals、compareTo和toString方法
你曾經(jīng)查看過(guò)java文檔中的Object類嗎?也許吧。每當(dāng)你向上追溯繼承樹的時(shí)候都會(huì)止步于這個(gè)類。你會(huì)注意到,該類有幾個(gè)方法是每一個(gè)類都必須繼承的。而你最喜歡重寫的方法可能就是toString(), .equals() and .hashCode() 這三個(gè)了。(至于為何總是應(yīng)該同時(shí)重寫后兩個(gè)方法,請(qǐng)看Per-Åke Minborg寫的這篇文章:https://minborgsjavapot.blogspot.com/2014/10/new-java-8-object-support-mixin-pattern.html)
但是僅僅有這幾個(gè)方法顯然是不夠的。許多人將標(biāo)準(zhǔn)庫(kù)中的其他的接口如Comparable和Serializable加以組合。但是這樣真的明智嗎?為什么每個(gè)人都很迫切地去自己實(shí)現(xiàn)這些方法呢?事實(shí)上,當(dāng)你準(zhǔn)備將對(duì)象存儲(chǔ)在一些容器中,如HashMap,并且想要控制哈希沖突的時(shí)候,實(shí)現(xiàn)你自己的.equals()方法和.hashCode()方法確實(shí)有它的意義,但實(shí)現(xiàn)compareTo()和toString()方法又是為何呢?
本篇文章中我將提出一種使用到Speedment 開源項(xiàng)目上的軟件設(shè)計(jì)方法,這里的對(duì)象的方法被定義為存儲(chǔ)于變量上的方法引用,而不是重寫它們。這樣做確有一些好處:你的POJO將會(huì)更短小簡(jiǎn)潔,通用的方法可以不需要繼承而進(jìn)行復(fù)用并且你可以因地制宜地使用它們。
原始的代碼
首先我們來(lái)看下面的代碼:這里有一個(gè)典型的Java類Person。在使用中需要從一個(gè)Set中打印出每一個(gè)person對(duì)象,并且按照姓在前和名在后的順序排列(以防出現(xiàn)兩個(gè)相同姓氏的人)。
// Person.java public class Person implements Comparable<Person> { private final String firstname; private final String lastname; public Person(String firstname, String lastname) { this.firstname = firstname; this.lastname = lastname; } public String getFirstname() { return firstname; } public String getLastname() { return lastname; } @Override public int hashCode() { int hash = 7; hash = 83 * hash + Objects.hashCode(this.firstname); hash = 83 * hash + Objects.hashCode(this.lastname); return hash; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; final Person other = (Person) obj; if (!Objects.equals(this.firstname, other.firstname)) { return false; } return Objects.equals(this.lastname, other.lastname); } @Override public int compareTo(Person that) { if (this == that) return 0; else if (that == null) return 1; int comparison = this.firstname.compareTo(that.firstname); if (comparison != 0) return comparison; comparison = this.lastname.compareTo(that.lastname); return comparison; } @Override public String toString() { return firstname + " " + lastname; } }
// Main.java public class Main { public static void main(String... args) { final Set people = new HashSet<>(); people.add(new Person("Adam", "Johnsson")); people.add(new Person("Adam", "Samuelsson")); people.add(new Person("Ben", "Carlsson")); people.add(new Person("Ben", "Carlsson")); people.add(new Person("Cecilia", "Adams")); people.stream() .sorted() .forEachOrdered(System.out::println); } }
Output
run:
Adam Johnsson
Adam Samuelsson
Ben Carlsson
Cecilia Adams
BUILD SUCCESSFUL (total time: 0 seconds)
Person
類實(shí)現(xiàn)了一些方法來(lái)控制輸出。 hashCode()
和equals()
方法確保同一個(gè)person對(duì)象不會(huì)被重復(fù)添加到set中。.compareTo()
方法用于排序方法中生成應(yīng)有的順序。而重寫方法toString()
是在System.out.println()
被調(diào)用的時(shí)候控制每個(gè)Person對(duì)象的輸出格式。你認(rèn)出這種結(jié)構(gòu)了嗎?幾乎任何一個(gè)java工程中都會(huì)有它。
替代這些代碼
相比于將所有這些方法寫入Person類中,我們可以讓它保持盡量的簡(jiǎn)潔,使用方法引用去處理它們。我們可以刪除所有equals(),hashCode(),compareTo()和toString()的樣板式代碼,取而代之的是下面介紹的兩個(gè)靜態(tài)變量:COMPARATOR
和TO_STRING
。
// Person.java public class Person { private final String firstname; private final String lastname; public Person(String firstname, String lastname) { this.firstname = firstname; this.lastname = lastname; } public String getFirstname() { return firstname; } public String getLastname() { return lastname; } public final static Comparator<Person> COMPARATOR = Comparator.comparing(Person::getFirstname) .thenComparing(Person::getLastname); public final static Function<Person, String> TO_STRING = p -> p.getFirstname() + " " + p.getLastname(); }
// Main.java public class Main { public static void main(String... args) { final Set people = new TreeSet<>(Person.COMPARATOR); people.add(new Person("Adam", "Johnsson")); people.add(new Person("Adam", "Samuelsson")); people.add(new Person("Ben", "Carlsson")); people.add(new Person("Ben", "Carlsson")); people.add(new Person("Cecilia", "Adams")); people.stream() .map(Person.TO_STRING) .forEachOrdered(System.out::println); } }
Output
run:
Adam Johnsson
Adam Samuelsson
Ben Carlsson
Cecilia Adams
BUILD SUCCESSFUL (total time: 0 seconds)
這樣實(shí)現(xiàn)的好處是我們可以在不用更改Person類的情況下替換排序策略或打印格式。這將使代碼擁有更強(qiáng)的可維護(hù)性和復(fù)用性,更不用說(shuō)更快的編寫速度了。
譯文鏈接:http://www.codeceo.com/article/java-equals-compareto-tostring.html
英文原文:Get Rid of Equals, CompareTo and toString
以上就是Java中替代equals,compareTo和toString的方法的詳細(xì)內(nèi)容,更多關(guān)于Java替代equals,compareTo和toString的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- Java中compareTo方法使用小結(jié)
- Java中BigDecimal的equals方法和compareTo方法的區(qū)別詳析
- Java中BigDecimal比較大小的3種方法(??compareTo()、??equals()??和??compareTo()??)
- Java compareTo用法詳解
- JavaSE中compare、compareTo的區(qū)別
- Java 基礎(chǔ):string中的compareTo方法
- java compare compareTo方法區(qū)別詳解
- java compareTo和compare方法比較詳解
- java使用compareTo實(shí)現(xiàn)一個(gè)類的對(duì)象之間比較大小操作
- Java中比較運(yùn)算符compareTo()、equals()與==的區(qū)別及應(yīng)用總結(jié)
- 詳解java中保持compareTo和equals同步
- JAVA中compareTo方法的使用小結(jié)
相關(guān)文章
使用cmd根據(jù)WSDL網(wǎng)址生成java客戶端代碼的實(shí)現(xiàn)
這篇文章主要介紹了使用cmd根據(jù)WSDL網(wǎng)址生成java客戶端代碼的實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-03-03Java在創(chuàng)建文件時(shí)指定編碼的實(shí)現(xiàn)方法
本文主要介紹了Java在創(chuàng)建文件時(shí)指定編碼的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07Java實(shí)現(xiàn)簡(jiǎn)單學(xué)生管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)簡(jiǎn)單學(xué)生管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-07-07java中如何對(duì)arrayList按數(shù)字大小逆序排序
這篇文章主要介紹了java中如何對(duì)arrayList按數(shù)字大小逆序排序問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-04-04mybatis-plus?插入修改配置默認(rèn)值的實(shí)現(xiàn)方式
這篇文章主要介紹了mybatis-plus?插入修改配置默認(rèn)值的實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07Hibernate一對(duì)多關(guān)聯(lián)雙向關(guān)聯(lián)代碼實(shí)現(xiàn)分享
Hibernate一對(duì)多關(guān)聯(lián)雙向關(guān)聯(lián)代碼實(shí)現(xiàn)分享,大家參考使用吧2013-12-12SpringMVC的Body參數(shù)攔截的問(wèn)題
SpringMVC對(duì)出參和入?yún)⒂蟹浅S押玫耐卣怪С?方便你對(duì)數(shù)據(jù)的輸入和輸出有更大的執(zhí)行權(quán),我們?nèi)绾瓮ㄟ^(guò)SpringMVC定義的結(jié)果做一系列處理呢,需要的朋友可以參考下2018-06-06Intellij idea下使用不同tomcat編譯maven項(xiàng)目的服務(wù)器路徑方法詳解
今天小編就為大家分享一篇關(guān)于Intellij idea下使用不同tomcat編譯maven項(xiàng)目的服務(wù)器路徑方法詳解,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2019-02-02