溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java中如何替代equals,compareTo和toString

發(fā)布時(shí)間:2020-07-18 15:53:57 來源:億速云 閱讀:318 作者:小豬 欄目:編程語言

這篇文章主要為大家展示了Java中如何替代equals,compareTo和toString,內(nèi)容簡(jiǎn)而易懂,希望大家可以學(xué)習(xí)一下,學(xué)習(xí)完之后肯定會(huì)有收獲的,下面讓小編帶大家一起來看看吧。

我們都曾在POJO中重寫過equals(),compareTo()和toString()方法。但是另有其他能做到職責(zé)分離的更好的方法并帶來更簡(jiǎn)潔的代碼。閱讀這篇文章來一探究竟吧!

更簡(jiǎn)明的職責(zé)——擺脫equals、compareTo和toString方法

你曾經(jīng)查看過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)庫中的其他的接口如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ù)用并且你可以因地制宜地使用它們。

原始的代碼

首先我們來看下面的代碼:這里有一個(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)了一些方法來控制輸出。 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)變量:COMPARATORTO_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ù)用性,更不用說更快的編寫速度了。

以上就是關(guān)于Java中如何替代equals,compareTo和toString的內(nèi)容,如果你們有學(xué)習(xí)到知識(shí)或者技能,可以把它分享出去讓更多的人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI