Java自定义比较器实现中文排序

 更新时间:2020年08月21日 11:01:40   作者:刻剑求舟  
这篇文章主要介绍了Java自定义比较器实现中文排序,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧

compareTo 方法   

compareTo()是两个字符串对象比较大小,返回一个整数值,如果调用字符串对象大,返回正整数,反之,返回负整数。相等则返回0。compareTo()是两个字符串对象按ASCII比较大小(汉字是Unicode),返回一个整数值,如果调用字符串对象大,返回正整数,反之,返回负整数。相等则返回0。

Comparator 比较器   

Java 内实现自定义比较器比较简单,实现Comparator接口的compare()这个方法来制定排序规则,按照Java规范应满足以下约定,否则会抛Comparison method violates its general contract 异常。规则如下:

同时应满足以下约定:

自反性 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)),这点规范中原文是“not strictly required”,不是必须的,但是实现者应该知道不一致的后果,所以尽量实现这一要求.

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编码,所以排序时不是我们习惯用的拼音字母。如果还是刚才的实现,代码如下:

@Test
public void testCompareCN() {
 List<String> list = new ArrayList<>();
 list.add("中国");// 中->20013 unicode编码的4E2D
 list.add("英国");// 英-->33521 unicode编码的82F1
 list.add("美国");// 美->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);

 // 输出字符编码对应的十进制
 //char a = '美';
 //System.out.println((int) a);
}

输出的这个结果不符合我们的排序习惯,因此应该用Collator指定Locale.CHINA,代码应如下:

@Test
public void testCollator() {
 List<String> list = new ArrayList<>();
 list.add("中国");
 list.add("英国");
 list.add("美国");
 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,自定义比较器时要注意。

补充知识:Java 使用比较器对TreeSet进行自定义排序

比较器是个很方便的工具

一般定义格式为

public static class 类名 implements Comparator{
    @Override
    public int compare(Object o1, Object o2) {
      // TODO Auto-generated method stub
      return o1 - o2;//升序
      //return o2 - o1;降序
    }
}

对于一个类来说,比如图书类,定义一个比较器之后,就可以对图书类的价格属性进行排序,升序降序都可以。也可以对图书类的名字进行排序。

在创建集合类的时候传入一个比较器对象,系统就会识别比较器中的方法了。

例如:

TreeSet<Book> treeset = new TreeSet<Book>(new MyComparator());

下面是使用TreeSet集合+比较器对图书类价格实现的升序排序

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换个顺序即可
   */
  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("动物世界",50));
    treeset.add(new Book("时间简史",25));
    treeset.add(new Book("探索发现",60));
    treeset.add(new Book("恐龙时代",20));

    System.out.println(treeset);
  }
}

以上这篇Java自定义比较器实现中文排序就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 华为技术专家讲解JVM内存模型(收藏)

    华为技术专家讲解JVM内存模型(收藏)

    这篇文章主要介绍了华为技术专家讲解JVM内存模型(收藏)的相关知识,本文给大家介绍的非常详细,具有一定的收藏借鉴价值,需要的朋友可以参考下
    2021-05-05
  • Spring Boot缓存问题分析及解决方案

    Spring Boot缓存问题分析及解决方案

    SpringBoot提供缓存支持,提升应用性能,但可能出现缓存不一致、缓存穿透、缓存击穿等问题,分析了缓存基本概念、SpringBoot缓存支持、常见缓存问题及解决方案,感兴趣的朋友跟随小编一起看看吧
    2024-09-09
  • springboot工程如何使用阿里云OSS传输文件

    springboot工程如何使用阿里云OSS传输文件

    阿里云对象存储OSS(Object Storage Service)是一款海量、安全、低成本、高可靠的云存储服务,多种存储类型供选择,全面优化存储成本,非常适合存储非结构化数据,本文给大家介绍springboot工程使用阿里云OSS传输文件的操作,感兴趣的朋友一起看看吧
    2023-08-08
  • Mybatis单个参数的if判断报异常There is no getter for property named ''xxx'' in ''class java.lang.Integer''的解决方案

    Mybatis单个参数的if判断报异常There is no getter for property named ''x

    今天小编就为大家分享一篇关于Mybatis单个参数的if判断报异常There is no getter for property named 'xxx' in 'class java.lang.Integer'的解决方案,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • Java使用DateTimeFormatter实现格式化时间

    Java使用DateTimeFormatter实现格式化时间

    这篇文章主要介绍了Java使用DateTimeFormatter实现格式化时间,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-01-01
  • RestTemplate调用POST和GET请求示例详解

    RestTemplate调用POST和GET请求示例详解

    这篇文章主要为大家介绍了RestTemplate调用POST和GET请求示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • Spring MVC Controller返回值及异常的统一处理方法

    Spring MVC Controller返回值及异常的统一处理方法

    这篇文章主要给大家介绍了关于Spring MVC Controller返回值及异常的统一处理方法,文中通过示例代码介绍的非常详细,对大家的学习或者使用Spring MVC具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-11-11
  • SpringBoot 自动扫描第三方包及spring.factories失效的问题解决

    SpringBoot 自动扫描第三方包及spring.factories失效的问题解决

    这篇文章主要介绍了SpringBoot 自动扫描第三方包及spring.factories失效的问题,本文给大家分享最新解决方法,需要的朋友可以参考下
    2023-05-05
  • java实现左旋转字符串

    java实现左旋转字符串

    这篇文章主要为大家详细介绍了java实现左旋转字符串,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-03-03
  • 关于stream().sorted()以及java中常用的比较器排序

    关于stream().sorted()以及java中常用的比较器排序

    这篇文章主要介绍了关于stream().sorted()以及java中常用的比较器排序,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-05-05

最新评论