Java自定義比較器實(shí)現(xiàn)中文排序
compareTo 方法
compareTo()是兩個(gè)字符串對(duì)象比較大小,返回一個(gè)整數(shù)值,如果調(diào)用字符串對(duì)象大,返回正整數(shù),反之,返回負(fù)整數(shù)。相等則返回0。compareTo()是兩個(gè)字符串對(duì)象按ASCII比較大小(漢字是Unicode),返回一個(gè)整數(shù)值,如果調(diào)用字符串對(duì)象大,返回正整數(shù),反之,返回負(fù)整數(shù)。相等則返回0。
Comparator 比較器
Java 內(nèi)實(shí)現(xiàn)自定義比較器比較簡(jiǎn)單,實(shí)現(xiàn)Comparator接口的compare()這個(gè)方法來(lái)制定排序規(guī)則,按照J(rèn)ava規(guī)范應(yīng)滿(mǎn)足以下約定,否則會(huì)拋Comparison method violates its general contract 異常。規(guī)則如下:
同時(shí)應(yīng)滿(mǎn)足以下約定:
自反性 sgn(compare(x, y)) == -sgn(compare(y, x))
傳遞性 compare(x, y) > 0 compare(y, z)>0) =>得出 compare(x, z)>0
一致性 (compare(x, y)==0) == (x.equals(y)),這點(diǎn)規(guī)范中原文是“not strictly required”,不是必須的,但是實(shí)現(xiàn)者應(yīng)該知道不一致的后果,所以盡量實(shí)現(xiàn)這一要求.
Comparator<String> comparator = new Comparator<String>() { @Override public int compare(String s1, String s2) { return s1.compareTo(s2); } };
以下代碼示例:
@Test public void testCompare() { List<String> list = new ArrayList<>(); list.add("java"); list.add("php"); list.add("c++"); System.out.println("排序前-->" + list); Comparator<String> comparator = new Comparator<String>() { @Override public int compare(String s1, String s2) { return s1.compareTo(s2); } }; Collections.sort(list, comparator); System.out.println("排序后-->" + list); Collections.reverse(list); System.out.println("排序后逆序-->" + list); }
Comparator中文排序
中文漢字是Unicode編碼,所以排序時(shí)不是我們習(xí)慣用的拼音字母。如果還是剛才的實(shí)現(xiàn),代碼如下:
@Test public void testCompareCN() { List<String> list = new ArrayList<>(); list.add("中國(guó)");// 中->20013 unicode編碼的4E2D list.add("英國(guó)");// 英-->33521 unicode編碼的82F1 list.add("美國(guó)");// 美->32654 unicode編碼的7F8E // 漢字unicode編碼表 http://www.chi2ko.com/tool/CJK.htm System.out.println("排序前-->" + list); Comparator<String> comparator = new Comparator<String>() { @Override public int compare(String s1, String s2) { int b = s1.compareTo(s2); return b; } }; Collections.sort(list, comparator); System.out.println("排序后-->" + list); Collections.reverse(list); System.out.println("排序后逆序-->" + list); // 輸出字符編碼對(duì)應(yīng)的十進(jìn)制 //char a = '美'; //System.out.println((int) a); }
輸出的這個(gè)結(jié)果不符合我們的排序習(xí)慣,因此應(yīng)該用Collator指定Locale.CHINA,代碼應(yīng)如下:
@Test public void testCollator() { List<String> list = new ArrayList<>(); list.add("中國(guó)"); list.add("英國(guó)"); list.add("美國(guó)"); System.out.println("排序前-->" + list); Collections.sort(list, new Comparator<String>() { @Override public int compare(String s1, String s2) { String o1 = ""; String o2 = ""; if (s1 != null) { o1 = s1; } if (s2 != null) { o2 = s2; } Collator instance = Collator.getInstance(Locale.CHINA); return instance.compare(o1, o2); } }); System.out.println("排序后-->" + list); Collections.reverse(list); System.out.println("排序后逆序-->" + list); }
值得注意的是,compareTo不能傳入null,自定義比較器時(shí)要注意。
補(bǔ)充知識(shí):Java 使用比較器對(duì)TreeSet進(jìn)行自定義排序
比較器是個(gè)很方便的工具
一般定義格式為
public static class 類(lèi)名 implements Comparator{ @Override public int compare(Object o1, Object o2) { // TODO Auto-generated method stub return o1 - o2;//升序 //return o2 - o1;降序 } }
對(duì)于一個(gè)類(lèi)來(lái)說(shuō),比如圖書(shū)類(lèi),定義一個(gè)比較器之后,就可以對(duì)圖書(shū)類(lèi)的價(jià)格屬性進(jìn)行排序,升序降序都可以。也可以對(duì)圖書(shū)類(lèi)的名字進(jìn)行排序。
在創(chuàng)建集合類(lèi)的時(shí)候傳入一個(gè)比較器對(duì)象,系統(tǒng)就會(huì)識(shí)別比較器中的方法了。
例如:
TreeSet<Book> treeset = new TreeSet<Book>(new MyComparator());
下面是使用TreeSet集合+比較器對(duì)圖書(shū)類(lèi)價(jià)格實(shí)現(xiàn)的升序排序
package test; import java.util.Comparator; import java.util.TreeSet; public class 比較器的使用 { public static class Book{ String name; int price; public Book(String name, int price) { this.name = name; this.price = price; } @Override public String toString() { // TODO Auto-generated method stub return "Book:" + name + " Price:" + price; } } /** * * @author Administrator * 升序比較器,降序只要將b1,b2換個(gè)順序即可 */ public static class MyComparator implements Comparator{ @Override public int compare(Object o1, Object o2) { // TODO Auto-generated method stub Book b1 = (Book) o1; Book b2 = (Book) o2; return b1.price - b2.price; } } public static void main(String[] args) { TreeSet<Book> treeset = new TreeSet<Book>(new MyComparator()); treeset.add(new Book("動(dòng)物世界",50)); treeset.add(new Book("時(shí)間簡(jiǎn)史",25)); treeset.add(new Book("探索發(fā)現(xiàn)",60)); treeset.add(new Book("恐龍時(shí)代",20)); System.out.println(treeset); } }
以上這篇Java自定義比較器實(shí)現(xiàn)中文排序就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
java Apache poi 對(duì)word doc文件進(jìn)行讀寫(xiě)操作
這篇文章主要介紹了Apache poi 對(duì)word doc文件進(jìn)行讀寫(xiě)操作的相關(guān)資料,需要的朋友可以參考下2017-01-01Java synchronized輕量級(jí)鎖的核心原理詳解
這篇文章主要為大家詳細(xì)介紹了Java synchronized輕量級(jí)鎖的核心原理,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-03-03VScode 打造完美java開(kāi)發(fā)環(huán)境最新教程
這篇文章主要介紹了VScode 打造完美java開(kāi)發(fā)環(huán)境最新教程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-12-12java中URLEncoder.encode與URLDecoder.decode處理url特殊參數(shù)的方法
這篇文章主要給大家介紹了關(guān)于java中URLEncoder.encode與URLDecoder.decode處理url特殊參數(shù)的方法,文中介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面來(lái)一起看看吧。2017-03-03SpringBoot實(shí)現(xiàn)PDF添加水印的示例
本文主要介紹了SpringBoot實(shí)現(xiàn)PDF添加水印的示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-05-05nacos在liunx系統(tǒng)中啟動(dòng)成功瀏覽器卻訪問(wèn)不了的解決方法
在linux下搭建nacos,現(xiàn)在想要啟動(dòng),訪問(wèn)nacos頁(yè)面,訪問(wèn)不了,所以本文小編將給大家介紹nacos在liunx系統(tǒng)中啟動(dòng)成功,瀏覽器卻訪問(wèn)不了?全面的解決辦法,需要的朋友可以參考下2023-09-09