IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> Java中的Comparable接口、Comparator接口与CompareTo方法 -> 正文阅读

[Java知识库]Java中的Comparable接口、Comparator接口与CompareTo方法

Java集合中的Comparable接口和Comparator接口的区别


Comparable接口

  • Comparable简介

Comparable接口位于java.lang.Comparable,如果一个类实现了Comparable接口,就说明这个类是支持排序的,那在实现Comparable接口的类的对象的List就可以通过调用Collections.sortArrays.sort进行排序。

  • Comparable接口的定义
public interface Comparable<T> {
    int compareTo(T t);
}
  • 测试演示

1.定义一个员工Employee类:

public class Employee {
    private String name;
    private int age;
    private double salary;

// getter和setter方法省略了 自己测试的话需要带上

    public Employee(String name, int age, double salary) {
        this.name = name;
        this.age = age;
        this.salary = salary;
    }

    public Employee() {
    }

    @Override
    public String toString() {
        return "Employee{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", salary=" + salary +
                '}';
    }
}

2.定义 EmployeeComparable类,继承Employee , 实现 Comparable接口,重写CompareTo方法。

public class EmployeeComparable extends Employee implements Comparable<Employee> {
    @Override
    public int compareTo(Employee o) {
        //需求: age年龄从小到大排列, salary工资从大到小排
        if (this.getAge() - o.getAge() == 0) {
            return (int) (o.getSalary() - this.getSalary());
        } else {
            return this.getAge() - o.getAge();
        }
    }

    public EmployeeComparable(String name, int age, double score) {
        super(name, age, score);
    }
}

3.写测试类,测试排序功能

public class EmployeeComparableTest {
    @Test
    public void test() {
        EmployeeComparable s1 = new EmployeeComparable("张三", 18, 3000);
        EmployeeComparable s2 = new EmployeeComparable("李四", 15, 2000);
        EmployeeComparable s3 = new EmployeeComparable("王五", 25, 5055);
        EmployeeComparable s4 = new EmployeeComparable("赵六", 25, 6066);

        List list = new ArrayList();
        list.add(s1);
        list.add(s2);
        list.add(s3);
        list.add(s4);

        // 排序前
        System.out.println(list);

        //调用sort方法
        Collections.sort(list);

        // 排序后
        System.out.println(list);
    }
}

4.测试结果
在这里插入图片描述

根据排序前后的输出,可以看到年龄是从小到大的,而工资是从大到小的,达到了预期的排序效果.

在应用中为了更好的判断,一般的写法是:

if (this.getAge()<o.getAge()){
            return -1;
        }else if(this.getAge()>o.getAge()){
            return 1;
        }else{
            return 0;
        }

Comparator接口

  • Comparator简介

Comparator接口位于java.util.Comparator,Comparator 是比较器接口,当我们需要控制某个类的次序,而这个类是不支持排序的(没有实现comparable接口),那我们用比较器来进行排序,这个比较器需要实现comparator接口,简而言之就是通过实现Comparator接口来建立一个比较器,从而通过比较器来对这个类进行排序.

  • Comparator接口定义
package java.util;

public interface Comparator<T> {

    int compare(T o1, T o2);

    boolean equals(Object obj);
}`

如果要实现Comparator接口,一定要实现compare(T o1, T o2)函数,但是可以不实现equals(Object obj)函数。
因为任何类,默认都是已经实现了equals(Object obj)的。 Java中的一切类都是继承于java.lang.Object,在Object.java中实现了equals(Object obj)函数,所以所有类相当于都是实现了`equals(Object obj)函数的。

int compare(T o1, T o2) 是“比较o1和o2的大小”。返回“负数”,说明“o1<o2”;返回“零”,意味着“o1等于o2”;返回“正数”,说明“o1>o2”.

  • 测试演示
    1.定义 EmployeeComparator类,继承Employee , 实现 Comparator接口,重写compare方法。
public class EmployeeComparator extends Employee implements Comparator<Employee> {
    @Override
    //年龄采用升序 年龄相同工资采用降序
    public int compare(Employee o1, Employee o2) {
        if (o1.getAge() == o2.getAge()) {
            return (int) (o2.getSalary() - o1.getSalary());
        } else {
            return o1.getAge() - o2.getAge();
        }
    }

    public EmployeeComparator() {
        super();
    }

    public EmployeeComparator(String name, int age, double salary) {
        super(name, age, salary);
    }
}

2.写测试类,测试排序功能

public class EmployeeComparatorTest {
    @Test
    public void test() {
        EmployeeComparator s1 = new EmployeeComparator("张三", 18, 3000);
        EmployeeComparator s2 = new EmployeeComparator("李四", 15, 2000);
        EmployeeComparator s3 = new EmployeeComparator("王五", 25, 5055);
        EmployeeComparator s4 = new EmployeeComparator("赵六", 25, 6066);

        List list = new ArrayList();
        list.add(s1);
        list.add(s2);
        list.add(s3);
        list.add(s4);

        // 排序前
        System.out.println(list);

        //调用sort方法
        Collections.sort(list, new EmployeeComparator());

        // 排序后
        System.out.println(list);
    }
}

3.测试结果
在这里插入图片描述

根据排序前后的输出,可以看到年龄是从小到大的,而工资是从大到小的,达到了预期的排序效果.


总结Comparable与comparator

  1. Comparable 位于 java.lang 包, Comparator 位于java.util 包。
  2. 实现Comparable接口需要重写compareTo()方法,实现Comparator方法需要重写compare()方法,这两个方法的类型都是int
  3. Comparable是排序接口,相当于内部比较器,称为自然排序,Comparator是比较器,相当于外部比较器,称为比较器排序,外部比较器可以对内部比较器进行扩充。
  4. int test = s1.compareTo(s2); result = 0 , 则 s1=s2 ; result < 0, 则 s1 0 , 则 s1 > s2 。int test= compare(T o1, T o2); 结果同上。 (若是 o2-o1,则反之。)
  5. 总而言之,如果对象的排序需要基于自然顺序,使用 Comparable比较合适,如果需要按照对象的不同属性进行排序,使用 Comparator比较合适。

拓展Java中的compareTo方法

  • 返回参与比较的前后两个字符串的asc码的差值,如果两个字符串首字母不同,则该方法返回首字母的asc码的差值
		//ASCII表中A是65 D是68
		String s1 = "A";
        String s2 = "D";        
        System.out.println(s1.compareTo(s2));
        //结果为-3
  • 如果首字符相同比较下一个字符,直到有不同的为止,同样返回asc码的差值
		String s1 = "AAAAA";
        String s2 = "AAAAD";        
        System.out.println(s1.compareTo(s2));
        //结果为-3
  • 如果两个字符不同,但是前面参照比较的相同,就返回两个字符串的长度差值
		String s1 = "AA";
        String s2 = "AA123456789";        
        System.out.println(a1.compareTo(a2));
        //结果为-9

返回正数说明s1>s2,返回负数说明s1<s2.

  • 数字类型不能用compareTo,直接用’>’ ‘<’ '='等等(但也可以通过转换成String类型来使用),但封装数据类型可以,DateStringInteger、或者其他的,可以直接使用compareTo()
		Integer s1 = 6;
        Integer s2 = 8;
        System.out.println(n1.compareTo(n2));
        //结果为-2
  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2021-08-14 13:53:43  更:2021-08-14 13:54:39 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 4:18:39-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码