SpringBoot进阶教程(七十六)多维度排序查询
在项目中经常能遇到,需要对某些数据集合进行多维度排序的需求。对于集合多条件排序解决方案也有很多,今天我们就介绍一种,思路大致是设置一个分值的集合,这个分值是按照需求来设定大小的,再根据分值的大小对集合排序。
v需求背景
我们来模拟一个需求,现在需要查询一个用户列表,该列表需要实现的排序优先级如下:
- 付费用户排在前,非付费用户排在后
- 付费用户中,排序优先级:同城市的>同省的>等级高的>活跃用户>不活跃用户>其他用户
- 非付费用户中,排序优先级:等级高的>同城市的>其他用户
v代码实现
package com.user; import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor; import java.util.ArrayList;
import java.util.List; /**
* @Author toutou
* @Date 2023/2/18
* @Des
*/
@Data
@AllArgsConstructor
@NoArgsConstructor
@Builder
public class User {
private Integer id; /**
* 等级
*/
private int grade; /**
* 所在省份id
*/
private int provinceId; /**
* 所在城市ID
*/
private int cityId; /**
* 是否是活跃用户,true活跃,false不活跃
*/
private boolean lively; /**
* 是否开通支付,true开通,false未开通
*/
private boolean pay; public static List<User> getTestUserList() {
List<User> list = new ArrayList<>();
list.add(User.builder().id(0).grade(1).provinceId(1).cityId(22).lively(false).pay(false).build());
list.add(User.builder().id(1).grade(1).provinceId(1).cityId(100).lively(true).pay(true).build());
list.add(User.builder().id(2).grade(3).provinceId(5).cityId(100).lively(true).pay(true).build());
list.add(User.builder().id(3).grade(2).provinceId(1).cityId(98).lively(true).pay(true).build());
list.add(User.builder().id(4).grade(2).provinceId(1).cityId(100).lively(true).pay(true).build());
list.add(User.builder().id(5).grade(2).provinceId(3).cityId(100).lively(true).pay(true).build());
list.add(User.builder().id(6).grade(2).provinceId(1).cityId(101).lively(true).pay(false).build());
list.add(User.builder().id(7).grade(1).provinceId(6).cityId(100).lively(false).pay(true).build());
list.add(User.builder().id(8).grade(1).provinceId(1).cityId(98).lively(true).pay(false).build());
list.add(User.builder().id(9).grade(1).provinceId(5).cityId(100).lively(true).pay(true).build());
return list;
}
}
为了便于调试,我们在user类中创建一个测试方法getTestUserList,以此来生成测试数据。
package com.util; import java.math.BigDecimal;
import java.util.List; /**
* @Author toutou
* @Date 2023/2/18
* @Des
*/
public class SortHelper {
/**
* 比较两个BigDecimal集合
* @param left
* @param right
* @return leftList<rightList返回-1;leftList=rightList返回0;leftList>rightList返回1;
*/
public static int compareBigDecimalList(List<BigDecimal> left, List<BigDecimal> right) {
int length = Math.max(left.size(), right.size());
for (int i = 0; i < length; i++) {
if (left.size() < i + 1) {
return -1;
}
if (right.size() < i + 1) {
return 1;
} int value = left.get(i).compareTo(right.get(i));
if (value != 0) {
return value;
}
}
return 0;
}
}
package com.util; import com.user.User;
import org.apache.commons.lang3.tuple.ImmutablePair; import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.stream.Collectors; import reactor.core.publisher.Flux; /**
* @Author toutou
* @Date 2023/2/18
* @Des
*/
public class Sort {
/**
* 当前用户所在城市id,所在省份id
* @param cityId
* @param provinceId
* @return
*/
public List<User> sortUserList(int cityId, int provinceId) {
// 获取初始化测试list数据
List<User> list = User.getTestUserList();
if(list == null || list.size() == 0){
return list;
} List<ImmutablePair<User, List<BigDecimal>>> userAndScore = new ArrayList<>();
for (User user : list){
// 初始化一个排序的分值list
List<BigDecimal> scoreList = Flux.range(0, 6).map(p -> BigDecimal.ZERO).collectList().block();
userAndScore.add(new ImmutablePair<>(user, scoreList));
if(user.isPay()){
// 付费用户排序,付费用户为1,非付费用户为2
scoreList.set(0, BigDecimal.valueOf(1)); if(user.getCityId() == cityId){
scoreList.set(1, BigDecimal.valueOf(1));
}else{
scoreList.set(1, BigDecimal.valueOf(2));
} if(user.getProvinceId() == provinceId){
scoreList.set(2, BigDecimal.valueOf(1));
}else{
scoreList.set(2, BigDecimal.valueOf(2));
} scoreList.set(3, BigDecimal.valueOf(-user.getGrade()));
if(user.isLively()){
scoreList.set(4, BigDecimal.valueOf(1));
}else{
scoreList.set(4, BigDecimal.valueOf(2));
} scoreList.set(5, BigDecimal.valueOf(-user.getId()));
}else{
scoreList.set(0, BigDecimal.valueOf(2));
scoreList.set(1, BigDecimal.valueOf(-user.getGrade()));
if(user.getCityId() == cityId){
scoreList.set(2, BigDecimal.valueOf(1));
}else{
scoreList.set(2, BigDecimal.valueOf(2));
} scoreList.set(3, BigDecimal.valueOf(-user.getId()));
}
} return userAndScore.stream().sorted(Comparator.comparing(p -> p.getValue(), SortHelper::compareBigDecimalList)).map(ImmutablePair::getLeft).collect(Collectors.toList());
}
}

v源码地址
https://github.com/toutouge/javademosecond/tree/master/hellolearn
作 者:请叫我头头哥
出 处:http://www.cnblogs.com/toutou/
关于作者:专注于基础平台的项目开发。如有问题或建议,请多多赐教!
版权声明:本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接。
特此声明:所有评论和私信都会在第一时间回复。也欢迎园子的大大们指正错误,共同进步。或者直接私信我
声援博主:如果您觉得文章对您有帮助,可以点击文章右下角【推荐】一下。您的鼓励是作者坚持原创和持续写作的最大动力!
#comment_body_3242240 { display: none }
SpringBoot进阶教程(七十六)多维度排序查询的更多相关文章
- SpringBoot进阶教程(七十四)整合ELK
在上一篇文章<SpringBoot进阶教程(七十三)整合elasticsearch >,已经详细介绍了关于elasticsearch的安装与使用,现在主要来看看关于ELK的定义.安装及使用 ...
- SpringBoot进阶教程(七十)SkyWalking
流行的APM(Application Performance Management工具有很多,比如Cat.Zipkin.Pinpoint.SkyWalking.优秀的监控工具还有很多,其它比如还有za ...
- SpringBoot进阶教程(二十六)整合Redis之共享Session
集群现在越来越常见,当我们项目搭建了集群,就会产生session共享问题.因为session是保存在服务器上面的.那么解决这一问题,大致有三个方案,1.通过nginx的负载均衡其中一种ip绑定来实现( ...
- SpringBoot进阶教程(七十五)数据脱敏
无论对于什么业务来说,用户数据信息的安全性无疑都是非常重要的.尤其是在数字经济大火背景下,数据的安全性就显得更加重要.数据脱敏可以分为两个部分,一个是DB层面,防止DB数据泄露,暴露用户信息:一个是接 ...
- SpringBoot进阶教程(二十九)整合Redis 发布订阅
SUBSCRIBE, UNSUBSCRIBE 和 PUBLISH 实现了 发布/订阅消息范例,发送者 (publishers) 不用编程就可以向特定的接受者发送消息 (subscribers). Ra ...
- SpringBoot进阶教程(五十九)整合Codis
上一篇博文<详解Codis安装与部署>中,详细介绍了codis的安装与部署,这篇文章主要介绍介绍springboot整合codis.如果之前看过<SpringBoot进阶教程(五十二 ...
- SpringBoot进阶教程(七十一)详解Prometheus+Grafana
随着容器技术的迅速发展,Kubernetes已然成为大家追捧的容器集群管理系统.Prometheus作为生态圈Cloud Native Computing Foundation(简称:CNCF)中的重 ...
- SpringBoot非官方教程 | 第二十六篇: sprinboot整合elk,搭建实时日志平台
转载请标明出处: 原文首发于https://www.fangzhipeng.com/springboot/2017/07/11/sprinboot25-elk/ 本文出自方志朋的博客 这篇文章主要介绍 ...
- SpringBoot进阶教程(二十八)整合Redis事物
Redis默认情况下,事务支持被禁用,必须通过设置setEnableTransactionSupport(true)为使用中的每个redistplate显式启用.这样做会强制将当前重新连接绑定到触发m ...
- SpringBoot进阶教程(二十五)整合Redis之@Cacheable、@CachePut、@CacheEvict的应用
在上一篇文章(<SpringBoot(二十四)整合Redis>)中,已经实现了Spring Boot对Redis的整合,既然已经讲到Cache了,今天就介绍介绍缓存注解.各家互联网产品现在 ...
随机推荐
- 安全测试之探索 windows 游戏扫雷
作者:京东工业 宛煜昕 扫雷游戏相信很多人都从小玩过,在那个电脑游戏并不多的时代,扫雷成为玩的热度蛮高的一款游戏之一,然而就在有一次,接触到了一次不寻常的扫雷过程,使得后来我也有了这个冲动,也来做一次 ...
- 官方文档 | 【JVM调优体系】「GC底层调优实战」XPocket为终结性能问题而生—开发指南
XPocket 用户文档 XPocket 是PerfMa为终结性能问题而生的开源的插件容器,它是性能领域的乐高,将定位或者解决各种性能问题的常见的Linux命令,JDK工具,知名性能工具等适配成各种X ...
- 动态开点线段树&线段树合并学习笔记
动态开点线段树 使用场景 \(4 \times n\) 开不下. 值域需要平移(有负数). 什么时候开点 显然,访问的节点不存在时(只会在修改递归时开点). trick 区间里面有负数时,\(mid ...
- 系统建模之UML用例视图
<用例视图> 1 用例图的目标 who「参与者」:确定谁要使用系统 what「功能」:他们使用系统做什么? 2 用例图-四大主要组件 2.1 参与者 参与者:与应用程序或系统进行交互的用户 ...
- sync.Pool:提高Go语言程序性能的关键一步
1. 简介 本文将介绍 Go 语言中的 sync.Pool并发原语,包括sync.Pool的基本使用方法.使用注意事项等的内容.能够更好得使用sync.Pool来减少对象的重复创建,最大限度实现对象的 ...
- 如何使用sms-activate解决短信验证码问题
目录 前言 第一步:注册sms-activate 第二步:找到我们需要的服务 第三步:使用服务 前言 最近有许多小伙伴私信我,由于他们的工作需要让我安利一款接码工具供他们使用,于是我在调研各大接码平台 ...
- 麻了,不要再动不动就BeanUtil.copyProperties!!!
前言 最近项目上要求升级一个工具包hutool的版本,以解决安全漏洞问题,这不升级还好,一升级反而捅出了更大的篓子,究竟是怎么回事呢? 事件回顾 我们项目原先使用的hutool版本是5.7.2,在代码 ...
- Appuploader证书申请教程
转载:http://kxdang.com/topic/appuploader/certification.html IOS证书制作教程 点击苹果证书 按钮 点击新增 输入证书密码,名称 这个密码不是账 ...
- Vue3 element-plus 下拉分页 select分页
由于用 input 实现下拉分页不太理想,转换了一个角度,用 select 实现,以下是具体实现(script-setup TS) script-setup <script lang=" ...
- ES的索引结构与算法解析
作者:京东物流 李洪吉 提到ES,大多数爱好者想到的都是搜索引擎,但是明确一点,ES不等同于搜索引擎.不管是谷歌.百度.必应.搜狗为代表的自然语言处理(NLP).爬虫.网页处理.大数据处理的全文搜索引 ...