Java中替代equals,compareTo和toString的方法
我們都曾在POJO中重寫過equals(),compareTo()和toString()方法。但是另有其他能做到職責分離的更好的方法并帶來更簡潔的代碼。閱讀這篇文章來一探究竟吧!
更簡明的職責——擺脫equals、compareTo和toString方法
你曾經(jīng)查看過java文檔中的Object類嗎?也許吧。每當你向上追溯繼承樹的時候都會止步于這個類。你會注意到,該類有幾個方法是每一個類都必須繼承的。而你最喜歡重寫的方法可能就是toString(), .equals() and .hashCode() 這三個了。(至于為何總是應(yīng)該同時重寫后兩個方法,請看Per-Åke Minborg寫的這篇文章:https://minborgsjavapot.blogspot.com/2014/10/new-java-8-object-support-mixin-pattern.html)
但是僅僅有這幾個方法顯然是不夠的。許多人將標準庫中的其他的接口如Comparable和Serializable加以組合。但是這樣真的明智嗎?為什么每個人都很迫切地去自己實現(xiàn)這些方法呢?事實上,當你準備將對象存儲在一些容器中,如HashMap,并且想要控制哈希沖突的時候,實現(xiàn)你自己的.equals()方法和.hashCode()方法確實有它的意義,但實現(xiàn)compareTo()和toString()方法又是為何呢?
本篇文章中我將提出一種使用到Speedment 開源項目上的軟件設(shè)計方法,這里的對象的方法被定義為存儲于變量上的方法引用,而不是重寫它們。這樣做確有一些好處:你的POJO將會更短小簡潔,通用的方法可以不需要繼承而進行復(fù)用并且你可以因地制宜地使用它們。
原始的代碼
首先我們來看下面的代碼:這里有一個典型的Java類Person。在使用中需要從一個Set中打印出每一個person對象,并且按照姓在前和名在后的順序排列(以防出現(xiàn)兩個相同姓氏的人)。
// 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 類實現(xiàn)了一些方法來控制輸出。 hashCode()和equals() 方法確保同一個person對象不會被重復(fù)添加到set中。.compareTo() 方法用于排序方法中生成應(yīng)有的順序。而重寫方法toString()是在System.out.println() 被調(diào)用的時候控制每個Person對象的輸出格式。你認出這種結(jié)構(gòu)了嗎?幾乎任何一個java工程中都會有它。
替代這些代碼
相比于將所有這些方法寫入Person類中,我們可以讓它保持盡量的簡潔,使用方法引用去處理它們。我們可以刪除所有equals(),hashCode(),compareTo()和toString()的樣板式代碼,取而代之的是下面介紹的兩個靜態(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)
這樣實現(xiàn)的好處是我們可以在不用更改Person類的情況下替換排序策略或打印格式。這將使代碼擁有更強的可維護性和復(fù)用性,更不用說更快的編寫速度了。
譯文鏈接:http://www.codeceo.com/article/java-equals-compareto-tostring.html
英文原文:Get Rid of Equals, CompareTo and toString
以上就是Java中替代equals,compareTo和toString的方法的詳細內(nèi)容,更多關(guān)于Java替代equals,compareTo和toString的資料請關(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實現(xiàn)一個類的對象之間比較大小操作
- Java中比較運算符compareTo()、equals()與==的區(qū)別及應(yīng)用總結(jié)
- 詳解java中保持compareTo和equals同步
- JAVA中compareTo方法的使用小結(jié)
相關(guān)文章
解決報錯:java:讀取jar包時出錯:error in opening zip 
文章總結(jié):解決Java讀取jar包時出錯的問題,通過下載源碼并刷新項目解決了問題,希望對大家有所幫助2024-11-11
Java數(shù)據(jù)類型實現(xiàn)自動與強制轉(zhuǎn)換的示例代碼
Java數(shù)據(jù)類型之間的轉(zhuǎn)換有自動轉(zhuǎn)換和強制類型轉(zhuǎn)換,這篇文章主要給大家介紹Java數(shù)據(jù)類型如何實現(xiàn)自動轉(zhuǎn)換與強制轉(zhuǎn)換,需要的朋友可以參考下2023-05-05
webservice實現(xiàn)springboot項目間接口調(diào)用與對象傳遞示例
本文主要介紹了webservice實現(xiàn)springboot項目間接口調(diào)用與對象傳遞示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-07-07
Spring Boot + Kotlin整合MyBatis的方法教程
前幾天由于工作需要,便開始學習了kotlin,java基礎(chǔ)扎實學起來也還算比較快,對于kotlin這個編程語言自然是比java有趣一些,下面這篇文章主要給大家介紹了關(guān)于Spring Boot + Kotlin整合MyBatis的方法教程,需要的朋友可以參考下。2018-01-01
在Spring Boot中實現(xiàn)HTTP緩存的方法
緩存是HTTP協(xié)議的一個強大功能,但由于某些原因,它主要用于靜態(tài)資源,如圖像,CSS樣式表或JavaScript文件。本文重點給大家介紹在Spring Boot中實現(xiàn)HTTP緩存的方法,感興趣的朋友跟隨小編一起看看吧2018-10-10
Spring Boot應(yīng)用事件監(jiān)聽示例詳解
這篇文章主要給大家介紹了關(guān)于Spring Boot應(yīng)用事件監(jiān)聽的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2018-12-12

