Java五十七:比较接口(Comparable、Comparator)_java实现比较接口-程序员宅基地

技术标签: java  idea  JavaSE基础/核心知识学习笔记  开发语言  

比较接口

导读
  1. 自然排序:如String类、包装类或自己写的类已经实现了Comparable接口,并重写了compareTo方法,那么直接调用Arrays.sort( ) 方法,就可以实现排序
  2. 定制排序:不想使用已经定义好的排序方法,可以使用Arrays.sort(T[] a, Comparator<? super T> c )方法,创建匿名内部实现类,根据实际情况,临时性的自定义排序顺序,以实现精确控制
00 引入​
  • Java中的对象,正常情况下,只能进行比较:== 或 != , 不能进行 > 或 < 的 比较。 但在开发场景中,我们需要对多个对象进行排序,即需要比较对象的大小。因此,API提供了​ 两个比较接口 Comparable 和 Comparator,将需要进行排序的接口的类实现两个接口中的任意一个,​ 并配合Arrays.sort( ) 方法,即可实现对对象的排序
一、Comparable接口
  1. 自然排序

    ① 像String、包装类等实现了Comparable接口,重写了compareTo( ) 方法,给出了比较两个对象大小的方式,方式是从小到大。
    实现Comparable接口的对象列表(和数组),可以通过Collections.sort或Arrays.sort进行自动排序。

        @Test
        public void test(){
          
            String[] str = new String[]{
          "CC","JJ","MM","DD","GG"};
            Arrays.sort(str);
            System.out.println(Arrays.toString(str));
            // [CC, DD, GG, JJ, MM]
        }
    
  2. 重写compareTo( ) 的规则:

    ① 如果当前对象this大于形参对象obj,则返回正整数;

    ② 如果当前对象this小于形参对象obj,则返回负整数;

    ③ 如果当前对象this等于形参对象obj,则返回 0

  3. 对于自定义类来说,如果需要排序,我们可以让自定义类实现Comparable接口,重写

    compareTo(Object obj)方法,在该方法中指明如何排序

    public class CompareTest {
          
       @Test
        public void test2(){
          
            Goods[] str = new Goods[5];
            str[0] = new Goods("lenovo",19);
            str[1] = new Goods("huawei",69);
            str[2] = new Goods("xiaomi",49);
            str[3] = new Goods("huawei",99);
            str[4] = new Goods("dell",99);
            Arrays.sort(str);
            System.out.println(Arrays.toString(str));
            /* 
              输出结果为:
              [{name='lenovo', price=19.0}, {name='xiaomi', price=49.0}, 
              {name='huawei', price=69.0}, {name='dell', price=99.0},
              {name='huawei', price=99.0}]   
             */
        }
    }
    class Goods implements Comparable{
          
        private String name;
        private double price;
    
        public Goods() {
          
        }
        public Goods(String name, double price){
          
            this.name = name;
            this.price = price;
        }
    
    
        @Override
        public String toString() {
          
            return "{" +
                    "name='" + name + '\'' +
                    ", price=" + price +
                    '}';
        }
    
        @Override
        public int compareTo(Object o) {
          
            System.out.println("-----------------");
            if (o instanceof Goods){
          
                Goods goods = (Goods)o;
                if (this.price > goods.price) {
          
                    return 1;
                }                
                if(this.price < goods.price) {
          
                    return -1;
                }
                return this.name.compareTo(goods.name);//将name属性也进行了排名
            }
            throw new RuntimeException("传入的数据类型不正确");
        }
    }
    
    
二、Comparator接口 - - - 定制排序
  1. 背景

    当元素的类型没有实现java.lang.Comparable接口而又不方便修改代码,或者实现了java.lang.Comparable接口的排序规则不适合当前的额操作,那么就可以考虑而用Comparator类型的对象来排序,强行对多个对象进行整体排序的比较

    也可以将Comaprator传递给sort方法(如Collections.sort 或 Arrays.sort),从而允许在排序顺序上实现精确控制

  2. 重写compare(Object o1 , Object o2)方法,比较o1和o2的大小

    ① 如果返回正整数,表示o1 > o2
    ② 如果返回负整数,表示o1 < o2
    ③ 如果返回 0 ,表示相等

  3. 配合使用Arrays.sort(T[] a, Comparator<? super T> c )方法,此处Comparator实现类对象只使用一次,一般写为匿名实现类对象new Comparator( ) { }

    @Test
        public void test3(){
          
            Goods[] str1 = new Goods[6];
            str1[0] = new Goods("lenovo",18);
            str1[1] = new Goods("huawei",68);
            str1[2] = new Goods("xiaomi",48);
            str1[3] = new Goods("huawei",98);
            str1[4] = new Goods("dell",98);
            str1[5] = new Goods("oppo",298);
            // 接口在匿名内部类(假象的所谓内部类)实例化现象
            Arrays.sort(str1, new Comparator() {
          
        @Override
            public int compare(Object o1, Object o2) {
                     
                if (o1 instanceof Goods && o2 instanceof Goods){
          
                    Goods g1 = (Goods) o1;
                    Goods g2 = (Goods) o2;
                    //按照名字从大到小排
                    if (g1.getName().equals(g2.getName())){
          
                        return -Double.compare(g1.getPrice(),g2.getPrice());
                    }else
                        return -g1.getName().compareTo(g2.getName());
                    }
                throw new RuntimeException("输入数据结构异常");
                }
            });          
            System.out.println(Arrays.toString(str1));
            /* 
             输出的排序结果
             [{name='xiaomi', price=48.0}, {name='oppo', price=298.0}, 
             {name='lenovo', price=18.0}, {name='huawei', price=98.0}, 
             {name='huawei', price=68.0}, {name='dell', price=98.0}]
             */
            
            
        }
    
三、比较
  • Comparable接口的方式一旦确定,保证Comparable接口的实现类的对象在任何位置都可以调用Arrays.sort( )方法进行大小比较
  • Comparator接口属于临时性的比较
四、总结:

只要涉及到对象排序,就是这两个比较类

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/N_ZSX/article/details/121412686

智能推荐

使用UmcFramework和unimrcpclient.xml连接多个SIP设置的配置指南及C代码示例

在多媒体通信领域,MRCP(Media Resource Control Protocol)协议被广泛用于控制语音识别和合成等媒体资源。UniMRCP是一个开源的MRCP实现,提供了客户端和服务端的库。UmcFramework是一个基于UniMRCP客户端库的示例应用程序框架,它帮助开发者快速集成和测试MRCP客户端功能。本文将详细介绍如何使用UmcFramework和unimrcpclient.xml配置文件连接到多个SIP设置,以及如何用C代码进行示例说明。

java.net.ProtocolException: Server redirected too many times (20)-程序员宅基地

文章浏览阅读3k次。报错:java.net.ProtocolException: Server redirected too many times (20)1.没有检查到cookie,一直循环重定向。解决:CookieHandler.setDefault(new CookieManager(null, CookiePolicy.ACCEPT_ALL));URL url = new URL(url); ..._java.net.protocolexception: server redirected too many times (20)

springboot启动报错 Failed to scan *****/derbyLocale_ja_JP.jar from classloader hierarchy_failed to scan from classloader hierarchy-程序员宅基地

文章浏览阅读4.1k次。问题这是部分报错信息2019-07-11 14:03:34.283 WARN [restartedMain][DirectJDKLog.java:175] - Failed to scan [file:/D:/repo/org/apache/derby/derby/10.14.2.0/derbyLocale_ja_JP.jar] from classloader hierarchyjava...._failed to scan from classloader hierarchy

MATLAB-ones函数_matlab中ones函数-程序员宅基地

文章浏览阅读2.8k次,点赞3次,收藏7次。在MATLAB中,ones函数用于创建一个指定大小的由1组成的矩阵或数组。_matlab中ones函数

解决PS等软件出现应用程序无法正常启动(0xc000007b)_photoshop应用程序无法正常启动0xc000007b。请单击“确认”关闭应用程序。-程序员宅基地

文章浏览阅读3.9w次,点赞2次,收藏9次。  在使用电脑办公过程中,安装应用程序时难免遇到无法安装或者无法正常启动的问题,这对我们使用电脑带来了诸多不便。那遇到应用程序无法正常启动的问题要如何解决呢?相信大家肯定都是十分疑问的,每次都是只能忍痛重新安装软件。今天,小编就和大家探讨下应用程序无法正常启动的解决方法,帮助大家排忧解难。0xc000007b电脑图解1  第一种方案:SFC检查系统完整性来尝试修复丢失文件  1、打开电脑搜索输入cmd.exe,选择以管理员身份运行,跳出提示框时选择继续。0xc000007b电脑图解2_photoshop应用程序无法正常启动0xc000007b。请单击“确认”关闭应用程序。

oracle介质恢复和实例恢复的异同-程序员宅基地

文章浏览阅读396次。1、概念 REDO LOG是Oracle为确保已经提交的事务不会丢失而建立的一个机制。实际上REDO LOG的存在是为两种场景准备的:实例恢复(INSTANCE RECOVERY);介质恢复(MEDIA RECOVERY)。 实例恢复的目的是在数据库发生故障时,确保BUFFER CACHE中的数据不会丢失,不会造成数据库的..._oracle 实例恢复和介质恢复

随便推点

轻松搭建CAS 5.x系列(5)-增加密码找回和密码修改功能-程序员宅基地

文章浏览阅读418次。概述说明CAS内置了密码找回和密码修改的功能; 密码找回功能是,系统会吧密码重置的连接通过邮件或短信方式发送给用户,用户点击链接后就可以重置密码,cas还支持预留密码重置的问题,只有回答对了,才可以重置密码;系统可配置密码重置后,是否自动登录; 密码修改功能是,用户登录后输入新密码即可完成密码修改。安装步骤`1. 首先,搭建好cas sso server您需要按..._修改cas默认用户密码

springcloud(七) feign + Hystrix 整合 、-程序员宅基地

文章浏览阅读141次。之前几章演示的熔断,降级 都是 RestTemplate + Ribbon 和RestTemplate + Hystrix ,但是在实际开发并不是这样,实际开发中都是 Feign 远程接口调用。Feign + Hystrix 演示:  eruka(略)order 服务工程:  pom.xml<?xml version="1.0" encoding="U..._this is order 服务工程

YOLOv7如何提高目标检测的速度和精度,基于优化算法提高目标检测速度-程序员宅基地

文章浏览阅读3.4k次,点赞35次,收藏43次。学习率是影响目标检测精度和速度的重要因素之一。合适的学习率调度策略可以加速模型的收敛和提高模型的精度。在YOLOv7算法中,可以使用基于余弦函数的学习率调度策略(Cosine Annealing Learning Rate Schedule)来调整学习率。

linux中进程退出函数:exit()和_exit()的区别_linux结束进程可以用哪些函数,它们之间有何区别?-程序员宅基地

文章浏览阅读4k次,点赞4次,收藏9次。 linux中进程退出函数:exit()和_exit()的区别(1)_exit()执行后立即返回给内核,而exit()要先执行一些清除操作,然后将控制权交给内核。(2)调用_exit函数时,其会关闭进程所有的文件描述符,清理内存以及其他一些内核清理函数,但不会刷新流(stdin, stdout, stderr ...). exit函数是在_exit..._linux结束进程可以用哪些函数,它们之间有何区别?

sqlserver55555_sqlserver把小数点后面多余的0去掉-程序员宅基地

文章浏览阅读134次。select 5000/10000.0 --想变成0.5select 5500/10000.0 --想变成0.55select 5550/10000.0 --想变成0.555select 5555/10000.0 --想变成0.5555其结果分别为:0.5000000 0.5500000 0.5550000 0.5555000一、如果想去掉数字5后面多余的0 ,需要转化一下:selec..._sql server 去小数 0

Angular6 和 RXJS6 的一些改动_angular6,requestoptions改成了什么-程序员宅基地

文章浏览阅读3.1k次。例一:import { Injectable } from '@angular/core';import { Observable } from 'rxjs';import { User } from "./model/User";import { map } from 'rxjs/operators';import { Http, Response, Headers, RequestOp..._angular6,requestoptions改成了什么

推荐文章

热门文章

相关标签