首页 > 编程语言 >《Java 基础篇》三:排序方式

《Java 基础篇》三:排序方式

时间:2024-10-01 11:11:38浏览次数:7  
标签:Goods Java name 方式 price 排序 public String

Author: ACatSmiling

Since: 2024-10-01

在 Java 中经常会涉及到对象数组等的排序问题,那么就涉及到对象之间的比较问题。

Java 实现对象排序的方式有两种:

  • 自然排序:java.lang.Comparable
  • 定制排序:java.util.Comparator

自然排序

自然排序:java.lang.Comparable 接口强行对实现它的每个类的对象进行整体排序,这种排序被称为类的自然排序。

实现 Comparable 接口的类必须重写 compareTo(Object obj),两个对象通过 compareTo(Object obj) 的返回值来比较大小。

重写compareTo(Object obj)的规则:如果当前对象 this 大于形参对象 obj,则返回正整数,如果当前对象 this 小于形参对象 obj,则返回负整数,如果当前对象 this 等于形参对象 obj,则返回零。

实现 Comparable 接口的对象列表或数组,可以通过Collections.sort()(针对集合)或Arrays.sort()(针对数组)进行自动排序。实现此接口的对象可以用作有序映射中的键或有序集合中的元素,无需指定比较器。示例:

public class Test {
    public static void main(String[] args) {
        // 集合排序
        List<String> list = new ArrayList<>();
        list.add("AA");
        list.add("VV");
        list.add("BB");
        list.add("AC");
        list.add("CC");
        list.add("EE");
        list.add("DE");
        for (String str : list) {
            System.out.print(str + " ");// AA VV BB AC CC EE DE
        }
        System.out.println();
        Collections.sort(list);
        for (String str : list) {
            System.out.print(str + " ");// AA AC BB CC DE EE VV
        }
        System.out.println();

        // 数组排序
        String[] strings = {"AA", "VV", "BB", "AC", "CC", "EE", "DE"};
        System.out.println(Arrays.toString(strings));// [AA, VV, BB, AC, CC, EE, DE]
        Arrays.sort(strings);
        System.out.println(Arrays.toString(strings));// [AA, AC, BB, CC, DE, EE, VV]
    }
}

对于类 C 的每一个 e1 和 e2 来说,当且仅当e1.compareTo(e2) == 0e1.equals(e2)具有相同的 boolean 值时,类 C 的自然排序才叫做与 equals 一致。建议(虽然不是必需的)最好使自然排序与 equals 一致。

Comparable 的典型实现:(默认都是从小到大排列的)

  • String 类:按照字符串中字符的 Unicode 值进行比较。
  • Character 类:按照字符的 Unicode 值来进行比较。
  • 数值类型对应的包装类以及 BigInteger 类、BigDecimal 类:按照它们对应的数值大小进行比较。
  • Boolean 类:true 对应的包装类实例大于 false 对应的包装类实例。
  • Date 类、Time 类等:后面的日期时间比前面的日期时间大。

对于自定义类来说,如果需要排序,我们可以让自定义类实现 Comparable 接口,并重写 compareTo(Object obj),在 compareTo(Object obj) 中,指明如何排序。示例:

public class Test {
    public static void main(String[] args) {
        Goods[] arr = new Goods[5];
        arr[0] = new Goods("lenovo", 34);
        arr[1] = new Goods("dell", 43);
        arr[2] = new Goods("xiaomi", 12);
        arr[3] = new Goods("huawei", 65);
        arr[4] = new Goods("microsoft", 43);

        // 排序前:[Goods{name='lenovo', price=34.0}, Goods{name='dell', price=43.0}, Goods{name='xiaomi', price=12.0}, Goods{name='huawei', price=65.0}, Goods{name='microsoft', price=43.0}]
        System.out.println("排序前:" + Arrays.toString(arr));
        Arrays.sort(arr);
        // 排序后:[Goods{name='xiaomi', price=12.0}, Goods{name='lenovo', price=34.0}, Goods{name='microsoft', price=43.0}, Goods{name='dell', price=43.0}, Goods{name='huawei', price=65.0}]
        System.out.println("排序后:" + Arrays.toString(arr));
    }
}

class Goods implements Comparable {
    private String name;
    private double price;

    public Goods() {
    }

    public Goods(String name, double price) {
        this.name = name;
        this.price = price;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public double getPrice() {
        return price;
    }

    public void setPrice(double price) {
        this.price = price;
    }

    @Override
    public String toString() {
        return "Goods{" +
                "name='" + name + '\'' +
                ", price=" + price +
                '}';
    }

    // 先按照价格从低到高进行排序,如果价格相同,再按照名称从高到低进行排序
    @Override
    public int compareTo(Object o) {
        if (o instanceof Goods) {
            Goods goods = (Goods) o;
            if (this.price > goods.price) {
                return 1;
            } else if (this.price < goods.price) {
                return -1;
            } else {
                return -this.name.compareTo(goods.name);
            }
            // return Double.compare(this.getPrice(), goods.getPrice());
        }
        throw new RuntimeException("传入的数据类型有误");
    }
}

定制排序

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

public class Test {
    public static void main(String[] args) {
        // 集合排序
        List<String> list = new ArrayList<>();
        list.add("AA");
        list.add("VV");
        list.add("BB");
        list.add("AC");
        list.add("CC");
        list.add("EE");
        list.add("DE");
        for (String str : list) {
            System.out.print(str + " ");// AA VV BB AC CC EE DE
        }
        System.out.println();
        // 不再以String本身默认的从小到大排序,而是从大到小排序
        Collections.sort(list, new Comparator<String>() {
            @Override
            public int compare(String o1, String o2) {
                return o2.compareTo(o1);
            }
        });
        for (String str : list) {
            System.out.print(str + " ");// VV EE DE CC BB AC AA
        }
        System.out.println();

        // 数组排序
        String[] strings = {"AA", "VV", "BB", "AC", "CC", "EE", "DE"};
        System.out.println(Arrays.toString(strings));// [AA, VV, BB, AC, CC, EE, DE]
        // 不再以String本身默认的从小到大排序,而是从大到小排序
        Arrays.sort(strings, new Comparator<String>() {
            @Override
            public int compare(String o1, String o2) {
                return -o1.compareTo(o2);
            }
        });
        System.out.println(Arrays.toString(strings));// [VV, EE, DE, CC, BB, AC, AA]
    }
}

重写compare(Object o1,Object o2),比较 o1 和 o2 的大小: 如果方法返回正整数,则表示 o1 大于 o2;如果返回 0,表示相等;返回负整数,表示 o1 小于 o2。

public class Test {
    public static void main(String[] args) {
        Goods[] arr = new Goods[5];
        arr[0] = new Goods("lenovo", 34);
        arr[1] = new Goods("dell", 43);
        arr[2] = new Goods("xiaomi", 12);
        arr[3] = new Goods("huawei", 65);
        arr[4] = new Goods("lenovo", 43);

        // 排序前:[Goods{name='lenovo', price=34.0}, Goods{name='dell', price=43.0}, Goods{name='xiaomi', price=12.0}, Goods{name='huawei', price=65.0}, Goods{name='lenovo', price=43.0}]
        System.out.println("排序前:" + Arrays.toString(arr));
        // 不以 Goods 本身的自然排序方式排序,更改为:按产品名称从低到高进行排序,如果名称相同,再按照价格从高到低进行排序
        Arrays.sort(arr, new Comparator<Goods>() {
            @Override
            public int compare(Goods o1, Goods o2) {
                if (!o1.getName().equals(o2.getName())) {
                    return o1.getName().compareTo(o2.getName());
                } else {
                    if (o1.getPrice() < o2.getPrice()) {
                        return 1;
                    } else if (o1.getPrice() > o2.getPrice()) {
                        return -1;
                    }
                }
                return 0;
            }
        });
        // 排序后:[Goods{name='dell', price=43.0}, Goods{name='huawei', price=65.0}, Goods{name='lenovo', price=43.0}, Goods{name='lenovo', price=34.0}, Goods{name='xiaomi', price=12.0}]
        System.out.println("排序后:" + Arrays.toString(arr));
    }
}

class Goods implements Comparable {
    private String name;
    private double price;

    public Goods() {
    }

    public Goods(String name, double price) {
        this.name = name;
        this.price = price;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public double getPrice() {
        return price;
    }

    public void setPrice(double price) {
        this.price = price;
    }

    @Override
    public String toString() {
        return "Goods{" +
                "name='" + name + '\'' +
                ", price=" + price +
                '}';
    }

    // 先按照价格从低到高进行排序,如果价格相同,再按照名称从高到低进行排序
    @Override
    public int compareTo(Object o) {
        if (o instanceof Goods) {
            Goods goods = (Goods) o;
            if (this.price > goods.price) {
                return 1;
            } else if (this.price < goods.price) {
                return -1;
            } else {
                return -this.name.compareTo(goods.name);
            }
            // return Double.compare(this.getPrice(), goods.getPrice());
        }
        throw new RuntimeException("传入的数据类型有误");
    }
}

可以将 Comparator 传递给sort(),比如:Collections.sort()Arrays.sort(),从而允许在排序顺序上实现精确控制。

还可以使用 Comparator 来控制某些数据结构(如有序 set 或有序映射)的顺序,或者为那些没有自然顺序的对象 collection 提供排序。

Comparable VS Comparator

二者的区别:

  • Comparable 接口的方式一旦指定,能够保证 Comparable 接口实现类的对象在任何位置都可以比较大小。
  • Comparator 接口属于临时性的比较,什么时候需要什么时候实现。

原文链接

https://github.com/ACatSmiling/zero-to-zero/blob/main/JavaLanguage/java.md

标签:Goods,Java,name,方式,price,排序,public,String
From: https://www.cnblogs.com/acatsmiling/p/18442759

相关文章

  • java计算机毕业设计本尚化妆品网站(开题+程序+论文)
    本系统(程序+源码)带文档lw万字以上 文末可获取一份本项目的java源码和数据库参考。系统程序文件列表开题报告内容研究背景随着电子商务的蓬勃发展和消费者对美妆产品需求的日益增长,化妆品电商网站逐渐成为人们购买化妆品的重要渠道。本尚化妆品网站作为一个新兴的化妆品电......
  • Java 死锁及避免讲解和案例示范
    在大型分布式系统中,死锁是一种常见但难以排查的并发问题。特别是在Java领域,死锁问题可能导致系统崩溃或卡顿。本文将以电商交易系统为例,详细讲解如何识别和避免Java程序中的死锁问题,确保系统高效运行。1.什么是死锁?死锁是指多个线程相互持有对方所需的资源,且都在等待......
  • 《Java 基础篇》一:入门
    Author:ACatSmilingSince:2024-09-30bit和byte计算机本质是一系列的电路开关。每个开关存在两种状态:开(on)和关(off)。如果电路是开的,它的值是1,如果电路是关的,它的值是0。一个0或者一个1存储为一个比特(bit),是计算机中最小的存储单位。计算机中最基本的存储单元是......
  • 《Java 基础篇》二:面向对象
    Author:ACatSmilingSince:2024-09-30概述面向过程(POP)与面向对象(OOP):二者都是一种思想,面向对象是相对于面向过程而言的。面向过程,强调的是功能行为,以函数为最小单位,考虑怎么做。面向对象,将功能封装进对象,强调具备了功能的对象,以类/对象为最小单位,考虑谁来做。面向对......
  • Java Stream介绍
    JavaStreamAPI是Java8引入的一项强大功能,旨在简化集合数据的处理。它允许开发者以更简洁和声明性的方式执行复杂的数据操作。以下是对JavaStreamAPI的详细介绍,包括其核心概念、常见操作、性能优化以及最佳实践。一核心概念1.1流(Stream):流是一种对数据序列的抽......
  • Java 中的 volatile和synchronized和 ReentrantLock区别讲解和案例示范
    在Java的并发编程中,volatile、synchronized和ReentrantLock是三种常用的同步机制。每种机制都有其独特的特性、优缺点和适用场景。理解它们之间的区别以及在何种情况下使用哪种机制,对提高程序的性能和可靠性至关重要。本文将详细探讨这三种机制的特性、使用场景及示例......
  • [Python手撕]部分排序
    classSolution:defsubSort(self,array:List[int])->List[int]:n=len(array)start,end=-1,-1#从前往后找到第一个逆序foriinrange(n-1):ifarray[i]>array[i+1]:start=i+1......
  • JavaScript笔记
    基操数据类型原始类型对象类型MapandSet流程控制函数及面向对象函数方法常用内部对象面向对象编程(OOP)操作BOM元素操作DOM元素(I)操作表单jQuery基操js作为一种脚本语言,可以嵌入到HTML页面中js是双标签,可以写多行,也可以写一行内部标签<script>......
  • 基于java+ssm的大中型企业职工信息化平台设计
    前言......
  • java计算机毕业设计“小世界”私人空间(开题+程序+论文)
    本系统(程序+源码)带文档lw万字以上 文末可获取一份本项目的java源码和数据库参考。系统程序文件列表开题报告内容研究背景在信息爆炸的时代,个人隐私与个性化空间的需求日益增长。随着社交媒体和数字化生活的普及,人们渴望在繁忙的虚拟世界中拥有一片属于自己的“小世界”。......