作者主頁:paper jie_博客
本文作者:大家好,我是paper jie,感謝你閱讀本文,歡迎一建三連哦。
本文錄入于《JAVA數(shù)據(jù)結(jié)構(gòu)》專欄,本專欄是針對于大學(xué)生,編程小白精心打造的。筆者用重金(時間和精力)打造,將javaSE基礎(chǔ)知識一網(wǎng)打盡,希望可以幫到讀者們哦。
其他專欄:《算法詳解》《C語言》《javaSE》等
內(nèi)容分享:本期將會分享java數(shù)據(jù)結(jié)構(gòu)中的對象比較
目錄
priorityQueue中如何插入對象
元素的比較
基本類型的比較
對象類型的直接比較
對象正確的比較方式
重寫equals方法
基于Comparble接口類的比較
基于比較器比較
集合框架中priorityQueue的比較方式
priorityQueue中如何插入對象
我們知道,優(yōu)先級隊列在插入元素時有一個要求:需要可以比較的對象才能插入。這里我們需要知道怎樣插入自定義類型對象:
比如我們插入這個student對象:
class student {
int age;
String name;
public student(int age, String name) {
this.age = age;
this.name = name;
}
}
public class Test {
public static void main(String[] args) {
PriorityQueue<student> priorityQueue = new PriorityQueue<>();
priorityQueue.offer(new student(12,"小豬佩奇"));
priorityQueue.offer(new student(12,"小豬喬治"));
}
在運行后發(fā)現(xiàn)它會報類型不兼容的異常,這是因為在堆中插入元素,為了滿足堆的性質(zhì),需要進行對象的比較,但是我們的student類型對象時不能直接比較的,所以會報錯
?
元素的比較
基本類型的比較
在Java中,基本類型的對象是可以直接進行比較大小的
class TestCompare {
public static void main(String[] args) {
int a = 10;
int b = 20;
System.out.println(a > b);
System.out.println(a < b);
System.out.println(a == b);
char c1 = 'A';
char c2 = 'B';
System.out.println(c1 > c2);
System.out.println(c1 < c2);
System.out.println(c1 == c2);
boolean b1 = true;
boolean b2 = false;
System.out.println(b1 == b2);
System.out.println(b1 != b2);
}
}
對象類型的直接比較
class Card {
public int rank; // 數(shù)值
public String suit; // 花色
public Card(int rank, String suit) {
this.rank = rank;
this.suit = suit;
}
}
public class TestPriorityQueue {
public static void main(String[] args) {
Card c1 = new Card(1, "?");
Card c2 = new Card(2, "?");
Card c3 = c1;
//System.out.println(c1 > c2); // 編譯報錯
System.out.println(c1 == c2); // 編譯成功 ----> 打印false,因為c1和c2指向的是不同對象
//System.out.println(c1 < c2); // 編譯報錯
System.out.println(c1 == c3); // 編譯成功 ----> 打印true,因為c1和c3指向的是同一個對象
}
}
這里我們知道,直接進行對象比較的是地址,只有相同才會返回true,不同就會報錯。但是這里為毛==可以比較呢?這就得提到Object類了,因為自定義類也會繼承Object類,這個類中提供了equal方法,==的情況下就是用的Object的equal方法。但是這個方式比較的就是引用對象的地址,沒有比較對象的內(nèi)容,這就頭疼了。
// Object中equal的實現(xiàn),可以看到:直接比較的是兩個引用變量的地址
public boolean equals(Object obj) {
return (this == obj);
}
對象正確的比較方式
重寫equals方法
class student {
int age;
String name;
public student(int age, String name) {
this.age = age;
this.name = name;
}
@Override
public boolean equals(Object obj) {
if(this == obj) {
return true;
}
if(obj == null || !(obj instanceof student)) {
return false;
}
student o = (student) obj;
return this.age == ((student) obj).age && this.name.equals(o.name);
}
}
如果指向一個對象,返回true
如果傳入的是null或者不是student,返回false
按照類的成員對象比較,只要成員對象相同就返回true
注意下調(diào)其他引用類型的比較也要調(diào)用equals
這里的缺陷就是:equals只能按照相等來比較,不能比較大小
基于Comparble接口類的比較
Comparable接口的源碼:
public interface Comparable<E> {
// 返回值:
// < 0: 表示 this 指向的對象小于 o 指向的對象
// == 0: 表示 this 指向的對象等于 o 指向的對象
// > 0: 表示 this 指向的對象大于 o 指向的對象
int compareTo(E o);
}
對用戶自定義類型,想要按照大小比較時,在定義類的時候,實現(xiàn)Comparable接口即可。然后在類中實現(xiàn)compareTo方法:
class student implements Comparable<student>{
int age;
String name;
public student(int age, String name) {
this.age = age;
this.name = name;
}
@Override
public int compareTo(student o) {
if(o == null) {
return 1;
}
return this.age - o.age;
}
}
基于比較器比較
用戶自定義比較器類,需要實現(xiàn)Comparator接口:
public interface Comparator<T> {
// 返回值:
// < 0: 表示 o1 指向的對象小于 o2 指向的對象
// == 0: 表示 o1 指向的對象等于 o2 指向的對象
// > 0: 表示 o1 指向的對象等于 o2 指向的對象
int compare(T o1, T o2);
}
這里要注意區(qū)分Comparable和Comparator接口
在自定義比較器類中重寫compare方法:
import java.util.Comparator;
class Card {
public int rank; // 數(shù)值
public String suit; // 花色
public Card(int rank, String suit) {
this.rank = rank;
this.suit = suit;
}
}
class CardComparator implements Comparator<Card> {
// 根據(jù)數(shù)值比較,不管花色
// 這里我們認為 null 是最小的
@Override
public int compare(Card o1, Card o2) {
if (o1 == o2) {
return 0;
} if
(o1 == null) {
return -1;
}
if (o2 == null) {
return 1;
}
return o1.rank - o2.rank;
}
public static void main(String[] args){
Card p = new Card(1, "?");
Card q = new Card(2, "?");
Card o = new Card(1, "?");
// 定義比較器對象
CardComparator cmptor = new CardComparator();
// 使用比較器對象進行比較
System.out.println(cmptor.compare(p, o)); // == 0,表示牌相等
System.out.println(cmptor.compare(p, q)); // < 0,表示 p 比較小
System.out.println(cmptor.compare(q, p)); // > 0,表示 q 比較大
}
}
這里使用Comparator需要導(dǎo)入java.util包
集合框架中priorityQueue的比較方式
集合框架中的PriorityQueue底層使用堆結(jié)構(gòu),因此其內(nèi)部的元素必須要能夠比大小PriorityQueue采用了:Comparble和Comparator兩種方式。?
1. Comparble是默認的內(nèi)部比較方式,如果用戶插入自定義類型對象時,該類對象必須要實現(xiàn)Comparble接口,并覆寫compareTo方法
2. 用戶也可以選擇使用比較器對象,如果用戶插入自定義類型對象時,必須要提供一個比較器類,讓該類實現(xiàn)Comparator接口并覆寫compare方法。?
JDK中的源碼:
// 用戶如果沒有提供比較器對象,使用默認的內(nèi)部比較,將comparator置為null
public PriorityQueue() {
this(DEFAULT_INITIAL_CAPACITY, null);
}
// 如果用戶提供了比較器,采用用戶提供的比較器進行比較
public PriorityQueue(int initialCapacity, Comparator<? super E> comparator) {
// Note: This restriction of at least one is not actually needed,
// but continues for 1.5 compatibility
if (initialCapacity < 1)
throw new IllegalArgumentException();
this.queue = new Object[initialCapacity];
this.comparator = comparator;
}
private void siftUp(int k, E x) {
if (comparator != null)
siftUpUsingComparator(k, x);
else
siftUpComparable(k, x);
}
private void siftUpComparable(int k, E x) {
Comparable<? super E> key = (Comparable<? super E>) x;
while (k > 0) {
int parent = (k - 1) >>> 1;
Object e = queue[parent];
if (key.compareTo((E) e) >= 0)
break;
queue[k] = e;
k = parent;
}
queue[k] = key;
}
@SuppressWarnings("unchecked")
private void siftUpUsingComparator(int k, E x) {
while (k > 0) {
int parent = (k - 1) >>> 1;
Object e = queue[parent];
if (comparator.compare(x, (E) e) >= 0)
break;
queue[k] = e;
k = parent;
}
queue[k] = x;
}
畫圖分析:文章來源:http://www.zghlxwxcb.cn/news/detail-715316.html
?文章來源地址http://www.zghlxwxcb.cn/news/detail-715316.html
到了這里,關(guān)于【數(shù)據(jù)結(jié)構(gòu)】Java對象的比較的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!