Java中 List 集合,通过 Stream 流进行排序总结

一、数据准备

cpp 复制代码
public class OrderTest {
    private String channelCode;
    private BigDecimal rate;
    // 省略 getter、setter、toString()、constructor
}
cpp 复制代码
		List<OrderTest> orderTestList = new ArrayList<>();
        OrderTest z09 = new OrderTest("Z09", new BigDecimal("7.6677"));
        OrderTest B2C = new OrderTest("B2C", new BigDecimal("5.6666"));
        OrderTest Z04 = new OrderTest("Z04", new BigDecimal("4.3137"));
        OrderTest ALL = new OrderTest("ALL", new BigDecimal("4.3137"));
        OrderTest ALL1 = new OrderTest("ALL", new BigDecimal("4.5137"));
        // 演示多字段排序使用
        OrderTest z091 = new OrderTest("Z09", new BigDecimal("7.6671"));
        // 演示给 null 值排序用
        OrderTest z092 = new OrderTest("Z09", null);
        OrderTest B2C1 = new OrderTest("B2C", new BigDecimal("5.6666"));
        OrderTest Z041 = new OrderTest("Z04", null);
        orderTestList.add(z09);
        orderTestList.add(z091);
        orderTestList.add(B2C);
        orderTestList.add(Z04);
        orderTestList.add(ALL);
        orderTestList.add(ALL1);

二、单字段排序

2.1、升序

cpp 复制代码
 list.stram().sorted(Comparator.Comparing(YourClass::Class's Field)
cpp 复制代码
System.out.println("----------------------------------------------");
        System.out.println("只按照 channelCode 升序排序:");
        List<OrderTest> channelAsc =
                orderTestList.stream()
                        .sorted(Comparator.comparing(OrderTest::getChannelCode))
                        .collect(Collectors.toList());
        channelAsc.forEach(System.out::println);

结果:

2.2、降序

cpp 复制代码
 list.stram().sorted(Comparator.Comparing(YourClass::Class's Field, Comparator.reverseOrder())
cpp 复制代码
System.out.println("----------------------------------------------");
        System.out.println("只按照 channelCode 降序排序:");
        List<OrderTest> channelDesc =
                orderTestList.stream()
                        .sorted(Comparator.comparing(OrderTest::getChannelCode, Comparator.reverseOrder()))
                        .collect(Collectors.toList());
        channelDesc.forEach(System.out::println);

结果:

二、多字段排序

cpp 复制代码
利用的是 thenComparing():
升序 thenComparing(YourClass::Class's Field),
降序 thenComparing(YourClass::Class's Field, Comparator.reverseOrder())。

注意: 使用 thenComparing(YourClass::Class's Field).reversed() 的时候要注意排序要求,
如果先按照 A 字段升序 B 字段升序的话,使用 reversed() 之后的结果是对 A 降序 B 降序。
cpp 复制代码
System.out.println("----------------------------------------------");
        System.out.println("先按照 channelCode 升序,再按照 rate 升序排序:");
        List<OrderTest> channelCodeAscRateAscList =
                orderTestList.stream()
                        .sorted(Comparator.comparing(OrderTest::getChannelCode)
                                .thenComparing(OrderTest::getRate))
                        .collect(Collectors.toList());
        channelCodeAscRateAscList.forEach(System.out::println);

结果

先按照 channelCode 将序,再按照 rate 升序将序,使用 reversed():

cpp 复制代码
System.out.println("----------------------------------------------");
        System.out.println("先按照 channelCode 将序,再按照 rate 将序排序,使用 reversed():");
        List<OrderTest> channelCodeAscRateAscWithReversedList =
                orderTestList.stream()
                        .sorted(Comparator.comparing(OrderTest::getChannelCode)
                                .thenComparing(OrderTest::getRate).reversed())
                        .collect(Collectors.toList());
        channelCodeAscRateAscWithReversedList.forEach(System.out::println);

结果

三、对 null 值处理

cpp 复制代码
 Comparator.nullsFirst(Comparator.reverseOrder())  -- null排在前面,reverseOrder是倒序,升序用naturalOrder
 Comparator.nullsLast(Comparator.reverseOrder())   -- null排在后面,reverseOrder是倒序,升序用naturalOrder
cpp 复制代码
orderTestList.add(new OrderTest(("Z09")));
        orderTestList.add(new OrderTest(("B2C")));
        orderTestList.add(new OrderTest(("Z04")));
        System.out.println("----------------------------------------------");
        System.out.println("先按照 channelCode 升序,再按照 rate 降序并且 null 值放前面排序:");
        List<OrderTest> channelCodeAscRateDescNullFirstList = orderTestList.stream()
                .sorted(Comparator.comparing(OrderTest::getChannelCode)
                        .thenComparing(OrderTest::getRate, Comparator.nullsFirst(Comparator.reverseOrder())))
                .collect(Collectors.toList());
        channelCodeAscRateDescNullFirstList.forEach(System.out::println);

结果

四、对排序字段个数不固定的情况,如何排序

需求:排序字段个数不确定,如何实现动态排序?

伪代码示例:

cpp 复制代码
		Comparator<SurgUnArrangeResponse> comparing = null;
        for (int i = 0; i < dictItemList.size(); i++) {
            String cd = dictItemList.get(i).getCd();
            if (i == 0) {
                // 首次排序
                comparing = getComparingByDictCd(cd, null);
            } else {
            	// 第二次排序及以上
                comparing = getComparingByDictCd(cd, comparing);
            }
        }
        // comparing 比较器,保存了排序的规则,responseList 是被排序的集合
		responseList = responseList.stream().sorted(comparing)
						.collect(Collectors.toList());
		
		// getComparingByDictCd 方法
		private Comparator<SurgUnArrangeResponse> getComparingByDictCd(String dictCd, 
Comparator<SurgUnArrangeResponse> result = null;
        switch (dictCd) {
            case "age":
                result = comparing == null ? Comparator.comparing(SurgUnArrangeResponse::getAge, Comparator.nullsLast(Comparator.reverseOrder())) :
                        comparing.thenComparing(SurgUnArrangeResponse::getAge, Comparator.nullsLast(Comparator.reverseOrder()));
                break;
                ....
        	}
        	return result;
		}
相关推荐
weixin_472339465 分钟前
高效处理大体积Excel文件的Java技术方案解析
java·开发语言·excel
小毛驴85032 分钟前
Linux 后台启动java jar 程序 nohup java -jar
java·linux·jar
DKPT1 小时前
Java桥接模式实现方式与测试方法
java·笔记·学习·设计模式·桥接模式
好奇的菜鸟3 小时前
如何在IntelliJ IDEA中设置数据库连接全局共享
java·数据库·intellij-idea
程序视点3 小时前
Window 10文件拷贝总是卡很久?快来试试这款小工具,榨干硬盘速度!
windows
wuk9983 小时前
基于MATLAB编制的锂离子电池伪二维模型
linux·windows·github
DuelCode4 小时前
Windows VMWare Centos Docker部署Springboot 应用实现文件上传返回文件http链接
java·spring boot·mysql·nginx·docker·centos·mybatis
优创学社24 小时前
基于springboot的社区生鲜团购系统
java·spring boot·后端
幽络源小助理4 小时前
SpringBoot基于Mysql的商业辅助决策系统设计与实现
java·vue.js·spring boot·后端·mysql·spring
猴哥源码4 小时前
基于Java+springboot 的车险理赔信息管理系统
java·spring boot