Java中按照不同字段进行排序

在Java中(默认Java8),假设有一个类定义为:

java 复制代码
class Pair{
	int a,b;
}

现在有一个Pair列表pairs,其类型为ArrayList<Pair>.

问题:(1)按照a字段从大到小排序并输出

(2)按照b字段从小到大排序并输出

说明:你可以对类进行一定改造,也可以借助库函数、库工具等进行实现。

此外,代码尽可能保持简洁,不要有太多冗余

1. 快速开发示例:Comparator+Lambda表达式

java 复制代码
import java.util.*;

class Pair{
	int a, b;
	public Pair(int a, int b){
		this.a = a;
		this.b = b;
	}
	// 方便输出观察
	@override
	public String toString(){
		return String.format("(%d, %d)", a, b);
	}
}

public class Main{
	public List<Pair> getPairs(){
		List<Pair> pairs = new ArrayList<>();
		// ...
	}
	public static void main(String[] args){
		// 构造一个 pairs 实例
		List<Pair> pairs = getPairs();
		// 按照字段 a 降序排列, 实现降序只需加负号即可
		pairs.sort(Comparator.compareInt(p->-p.a));
		System.out.println(pairs);
		pairs.sort(Comparator.compareInt(p->p.b));
		System.out.println(pairs);
	}
}

如上述示例所示,通过调用集合框架的sort方法并传入Comparator即可实现按照指定字段的排序,对于int类型,已经有一个Comparator.compareInt可以直接使用,内部使用Lambda表达式指定字段,对于需要降序输出的,添加负号即可。

2. 标准开发示例:Comparator+Getter

通常,按照JavaBean的设计思想,类的字段保持私有,如必要还需要设置为final类型以保证线程安全,对外暴露公共的getter方法,此时代码编写更加规范:

java 复制代码
import java.util.*;
class Pair{
	private final int a, b;
	public Pair(int a, int b){
		this.a=a;
		this.b=b;
	}
	public int getA(){return a;}
	public int getB(){return b;}
	public String toString(){return String.format("(%d,%d)", a, b);}
}
public class Main{
	public List<Pair> getPairs(){
		List<Pair> pairs = new ArrayList<>();
		// ...
	}
	public static void main(String[] args){
		// 构造一个 pairs 实例
		List<Pair> pairs = getPairs();
		// 按照字段 a 降序排列, Pair::getA表达式+reversed方法
		pairs.sort(Comparator.compareInt(Pair::getA).reversed());
		System.out.println(pairs);
		// 按照字段 b 升序排列
		pairs.sort(Comparator.compareInt(Pair::getB));
		System.out.println(pairs);
	}
}

3. 若字段a,b为String或者其他的自定义类型(已实现Comparable)

使用更通用的 comparing 方法

java 复制代码
// 按照 a 降序排列
pairs.sort(Comparator.comparing(Pair::getA).reversed());

// 按照 b 升序排列
pairs.sort(Comparator.comparing(Pair::getB));

```

此处看到getter的好处,在a,b类型变化时,排序代码基本保持不变。


### 4. 如何实现 Comparable
假设现在需求并不是通过不同的字段来进行灵活多样的排序,而是结合不同字段实现更复杂的排序行为,此时就需要使用 Comparable 接口,例如通过a+b实现升序排序,或者先通过a进行升序排序,当a相等时通过b升序排序等等,以后者为例,实现代码如下:
```java
import java.util.*;

class Pair implements Comparable<Pair>{
	int a, b;
	public Pair(int a, int b){
		this.a = a;
		this.b = b;
	}
	public int compareTo(Pair o){
		if (this.a==o.a){
			return this.b-o.b;
		}
		return this.a-o.a;
	}
}
// 排序代码
Collections.sort(pairs);
```
相关推荐
MinterFusion2 分钟前
如何在openKylin 2.0 SP2中安装Qt(v0.2.2)(上)
开发语言·qt·软件开发·系统维护·明德融创·openkylin
爱吃烤鸡翅的酸菜鱼3 分钟前
Java 事件发布-订阅机制全解析:从原生实现到主流中间件
java·中间件·wpf·事件·发布订阅
前端小D12 分钟前
JS模块化
开发语言·前端·javascript
无限码力12 分钟前
华为OD技术面真题 - JAVA开发- spring框架 - 7
java·开发语言·华为od·华为od面试真题·华为odjava八股文·华为odjava开发题目·华为odjava开发高频题目
05大叔17 分钟前
优化器Adam,神经网络处理文本,CNN,RNN
开发语言·python·机器学习
Lyyaoo.18 分钟前
【JAVA基础面经】JAVA中的异常
java·开发语言
ByteCraze20 分钟前
JavaScript 深拷贝完全指南:从入门到精通
开发语言·javascript·ecmascript
wenzhangli721 分钟前
ooderAgent 龙虾时代的统一认证体系
开发语言·php
一定要AK32 分钟前
JVM 全体系深度解析笔记
java·jvm·笔记
coder阿龙33 分钟前
基于SpringAI+Qdrant+Ollama本地模型和向量数据库开发问答和RAG检索
java·数据库·spring boot·ai·数据库开发