Java 多条件复杂排序小结
前言 今天下午做了拼多多在牛客网上的在线笔试题,感觉自己的智商被鄙视到了···不过其中一道题的某一部分引起了我极大的兴趣,感觉可以总结一下,做好积累~ 题目的部分我拍照如下所示
这里面最复杂的就是第3点,对秒杀活动进行排序,排序条件多,排序条件使用各不相同,在笔试中给我带来了很大的困扰,最后当然也是没做完啦····
解决方案 吃完晚饭开始寻找解决方法,觉得应该会很有意思,果然看到了比较好的做法,链接在这
java多条件优先级排序 — Comparator 这个方案以我这样几个月的菜鸟看来,最巧妙的就是把 多个比较器 Comparator放进一个 比较器列表 中,然后在需要时,在new一个比较器,然后在其中foreach使用各个比较器,而且注意看他代码24行的if,没有在循环中间返回结果为0的情况,而是在使用完所有比较器最后再返回0;这样就能按顺序依次使用各个比价器了。 那么回到我的题目中来
我先定义了4个比较器,分别用于4中不同项目的排序比较 //按人气从大到小对活动进行排序
private Comparator<Activity> renqiComparator = new Comparator<Activity>() {
@Override
public int compare(Activity o1, Activity o2) {
if(goods.get(o1.goodsId).renQi == goods.get(o2.goodsId).renQi) {
return 0;
} else {
return goods.get(o1.goodsId).renQi > goods.get(o2.goodsId).renQi ? -1 : 1;
}
}
}; //按商品id从小到大排序
private Comparator<Activity> idComparator = new Comparator<Activity>() {
@Override
public int compare(Activity o1, Activity o2) {
if(o1.goodsId == o2.goodsId) {
return 0;
} else {
return o1.goodsId > o2.goodsId ? 1 : -1;
}
}
}; //按活动开始时间从早到晚排序
private Comparator<Activity> startTimeComparator = new Comparator<Activity>() { @Override
public int compare(Activity o1, Activity o2) {
if(o1.startTime == o2.startTime) {
return 0;
} else {
return o1.startTime > o2.startTime ? 1 :-1;
}
}
}; //按活动的最后卖出时间从晚到早排序
private Comparator<Activity> sellTimeComparator = new Comparator<Activity>() { @Override
public int compare(Activity o1, Activity o2) {
if(o1.sellTime == o2.sellTime) {
return 0;
} else {
return o1.sellTime > o2.sellTime ? -1 : 1;
}
}
}; 然后定义了三个比较器列表,分别用于三种情况下的比较,然后在构造器中初始化这3个比较器列表 public Main3() {
//在构造器中把这个复杂的比较器列表进行初始化
//对于进行中(未售罄)的活动,按商品人气从高到低、商品ID从小到大排序
activityComparatorList1.add(renqiComparator);
activityComparatorList1.add(idComparator); //对于进行中(已售罄)的活动,按最后卖出时间从晚到早、商品人气从高到低、商品ID从小到大排序
activityComparatorList2.add(sellTimeComparator);
activityComparatorList2.add(renqiComparator);
activityComparatorList2.add(idComparator); //对于未开始的活动,依次按开始时间从早到晚、商品人气从高到低、商品ID从小到大排序
activityComparatorList3.add(startTimeComparator);
activityComparatorList3.add(renqiComparator);
activityComparatorList3.add(idComparator);
} 最后再需要对活动排序的地方这样使用,针对不同阶段使用不同的比较器列表。 Collections.sort(list, new Comparator<Activity>() { @Override
public int compare(Activity o1, Activity o2) { //在同一阶段内的比较
if(o1.limitQuantity>0 && o2.limitQuantity>0) {//进行中未售罄
for(Comparator<Activity> comparator : activityComparatorList1) {
if(comparator.compare(o1, o2) < 0) {
return -1;
} else if(comparator.compare(o1, o2) > 0) {
return 1;
}
}
return 0; //关键在这,里面的比较器给出0时不处理,指导全部比较完成都相等的情况下才给出0 } else if(o1.limitQuantity == 0 && o2.limitQuantity == 0) {//进行中已售罄
for(Comparator<Activity> comparator : activityComparatorList2) {
if(comparator.compare(o1, o2) < 0) {
return -1;
} else if(comparator.compare(o1, o2) > 0) {
return 1;
}
}
return 0; //关键在这,里面的比较器给出0时不处理,指导全部比较完成都相等的情况下才给出0 } else if(o1.startTime>time && o2.startTime>time) { //未开始活动
for(Comparator<Activity> comparator : activityComparatorList3) {
if(comparator.compare(o1, o2) < 0) {
return -1;
} else if(comparator.compare(o1, o2) > 0) {
return 1;
}
}
return 0; //关键在这,里面的比较器给出0时不处理,指导全部比较完成都相等的情况下才给出0
} //在不同阶段的比较
if(o1.limitQuantity > 0 && (o2.limitQuantity == 0 || o2.startTime>time)) {
return -1;
} else if(o1.limitQuantity == 0 && o2.startTime>time) {
return -1;
} else {
return 1;
} }
}); 这样我感觉是比较合理的解决了这样一个多重不同条件的比较问题,而且感觉没有使用过多的判断使得逻辑混乱。
附全部代码 这道题最后也没有放在OJ上测试过,所以可能有错,仅供参考 package com.pinduoduo; import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.List;
import java.util.Scanner; public class Main3 { public HashMap<Integer, Good> goods = new HashMap<>();
public ArrayList<Activity> activities = new ArrayList<>();
public List<Comparator<Activity>> activityComparatorList1 = new ArrayList<>();
public List<Comparator<Activity>> activityComparatorList2 = new ArrayList<>();
public List<Comparator<Activity>> activityComparatorList3 = new ArrayList<>(); //按人气从大到小对活动进行排序
private Comparator<Activity> renqiComparator = new Comparator<Activity>() {
@Override
public int compare(Activity o1, Activity o2) {
if(goods.get(o1.goodsId).renQi == goods.get(o2.goodsId).renQi) {
return 0;
} else {
return goods.get(o1.goodsId).renQi > goods.get(o2.goodsId).renQi ? -1 : 1;
}
}
}; //按商品id从小到大排序
private Comparator<Activity> idComparator = new Comparator<Activity>() {
@Override
public int compare(Activity o1, Activity o2) {
if(o1.goodsId == o2.goodsId) {
return 0;
} else {
return o1.goodsId > o2.goodsId ? 1 : -1;
}
}
}; //按活动开始时间从早到晚排序
private Comparator<Activity> startTimeComparator = new Comparator<Activity>() { @Override
public int compare(Activity o1, Activity o2) {
if(o1.startTime == o2.startTime) {
return 0;
} else {
return o1.startTime > o2.startTime ? 1 :-1;
}
}
}; //按活动的最后卖出时间从晚到早排序
private Comparator<Activity> sellTimeComparator = new Comparator<Activity>() { @Override
public int compare(Activity o1, Activity o2) {
if(o1.sellTime == o2.sellTime) {
return 0;
} else {
return o1.sellTime > o2.sellTime ? -1 : 1;
}
}
}; public Main3() {
//在构造器中把这个复杂的比较器列表进行初始化
//对于进行中(未售罄)的活动,按商品人气从高到低、商品ID从小到大排序
activityComparatorList1.add(renqiComparator);
activityComparatorList1.add(idComparator); //对于进行中(已售罄)的活动,按最后卖出时间从晚到早、商品人气从高到低、商品ID从小到大排序
activityComparatorList2.add(sellTimeComparator);
activityComparatorList2.add(renqiComparator);
activityComparatorList2.add(idComparator); //对于未开始的活动,依次按开始时间从早到晚、商品人气从高到低、商品ID从小到大排序
activityComparatorList3.add(startTimeComparator);
activityComparatorList3.add(renqiComparator);
activityComparatorList3.add(idComparator);
} public int addActivity(int startTime, int endTime, int goodsId, int limitQuantity) {
if(limitQuantity <= goods.get(goodsId).kuCun) {
Activity activity = new Activity();
activity.startTime = startTime;
activity.endTime = endTime;
activity.goodsId = goodsId;
activity.limitQuantity = limitQuantity;
activity.id = activities.size();
activities.add(activity);
return activity.id;
}
return -1;
} public int buyGoods(int time, int activityId, int quantity) {
Activity activity = activities.get(activityId);
int startTime = activity.startTime;
int endTime = activity.endTime;
int limitQuantity = activity.limitQuantity;
if(time < startTime || time >= endTime) {
return -1;
} else if(quantity > limitQuantity) {
return -1;
} else {
activity.limitQuantity -= quantity;
activity.sellTime = time;
return 0;
} } public List<Activity> getActivityList(int time) {
ArrayList<Activity> list = new ArrayList<>();
for (Activity activity : activities) {
if(time >= activity.startTime && time < activity.endTime) {
list.add(activity);
}
}
Collections.sort(list, new Comparator<Activity>() { @Override
public int compare(Activity o1, Activity o2) { //在同一阶段内的比较
if(o1.limitQuantity>0 && o2.limitQuantity>0) {//进行中未售罄
for(Comparator<Activity> comparator : activityComparatorList1) {
if(comparator.compare(o1, o2) < 0) {
return -1;
} else if(comparator.compare(o1, o2) > 0) {
return 1;
}
}
return 0; //关键在这,里面的比较器给出0时不处理,指导全部比较完成都相等的情况下才给出0 } else if(o1.limitQuantity == 0 && o2.limitQuantity == 0) {//进行中已售罄
for(Comparator<Activity> comparator : activityComparatorList2) {
if(comparator.compare(o1, o2) < 0) {
return -1;
} else if(comparator.compare(o1, o2) > 0) {
return 1;
}
}
return 0; //关键在这,里面的比较器给出0时不处理,指导全部比较完成都相等的情况下才给出0 } else if(o1.startTime>time && o2.startTime>time) { //未开始活动
for(Comparator<Activity> comparator : activityComparatorList3) {
if(comparator.compare(o1, o2) < 0) {
return -1;
} else if(comparator.compare(o1, o2) > 0) {
return 1;
}
}
return 0; //关键在这,里面的比较器给出0时不处理,指导全部比较完成都相等的情况下才给出0
} //在不同阶段的比较
if(o1.limitQuantity > 0 && (o2.limitQuantity == 0 || o2.startTime>time)) {
return -1;
} else if(o1.limitQuantity == 0 && o2.startTime>time) {
return -1;
} else {
return 1;
} }
});
return list;
} public static void main(String[] args) { Scanner sc = new Scanner(System.in);
Main3 main = new Main3();
int n = sc.nextInt();
int m = sc.nextInt();
for(int i=0; i<n; i++) {
Good good = new Good();
good.id = sc.nextInt();
good.renQi = sc.nextInt();
good.kuCun = sc.nextInt();
main.goods.put(good.id, good);
} sc.nextLine();
String[] ask = new String[m];
for(int i=0; i<m; i++) {
ask[i] = sc.nextLine();
}
for(int i=0; i<m; i++) {
String[] command = ask[i].split(" ");
if("add".equals(command[1])) {
int time = Integer.parseInt(command[0]);
int startTime = Integer.parseInt(command[2]);
int endTime = Integer.parseInt(command[3]);
int goodsId = Integer.parseInt(command[4]);
int limitQuantity = Integer.parseInt(command[5]);
System.out.println(main.addActivity(startTime, endTime, goodsId, limitQuantity)); } else if("buy".equals(command[1])) {
int time = Integer.parseInt(command[0]);
int activityId = Integer.parseInt(command[2]);
int quantity = Integer.parseInt(command[3]);
System.out.println(main.buyGoods(time, activityId, quantity)); } else if("list".equals(command[1])){
int time = Integer.parseInt(command[0]);
List<Activity> list = main.getActivityList(time);
for (Activity activity : list) {
System.out.print(activity.id+" ");
}
System.out.println();
}
}
} } class Good {
public int id;
public int renQi;
public int kuCun;
} class Activity {
public int id;
public int startTime;
public int endTime;
public int goodsId;
public int limitQuantity;
public int sellTime;
}
Java 多条件复杂排序小结的更多相关文章
- Java内存访问重排序笔记
>>关于重排序 重排序通常是编译器或运行时环境为了优化程序性能而采取的对指令进行重新排序执行的一种手段. 重排序分为两类:编译期重排序和运行期重排序,分别对应编译时和运行时环境. > ...
- Java使用条件语句和循环结构确定控制流
与任何程序设计语言一样,Java使用条件语句和循环结构确定控制流.本文将简单讲解条件.循环和switch. 一.块作用域 块(block),即复合语句.是指由一对大括号括起来的若干条简单的Java语句 ...
- JAVA 变量 数据类型 运算符 知识小结
---------------------------------------------------> JAVA 变量 数据类型 运算符 知识小结 <------------------ ...
- Java常见异常(Runtime Exception )小结(转)
原文链接:Java常见异常(Runtime Exception )小结 Java异常体系结构呈树状,其层次结构图如图 1所示: 本文重在Java中异常机制的一些概念.写本文的目的在 ...
- Java实现各种内部排序算法
数据结构中常见的内部排序算法: 插入排序:直接插入排序.折半插入排序.希尔排序 交换排序:冒泡排序.快速排序 选择排序:简单选择排序.堆排序 归并排序.基数排序.计数排序 直接插入排序: 思想:每次将 ...
- Java中List的排序和List的MAp
这里是一个类中类去实现条件优先排序的问题 package com.sun; import java.util.ArrayList; import java.util.Arrays; import ja ...
- java keytool证书工具使用小结
java keytool证书工具使用小结 在Security编程中,有几种典型的密码交换信息文件格式: DER-encoded certificate: .cer, .crt PEM-encod ...
- Java线性表的排序
Java线性表的排序 ——@梁WP 前言:刚才在弄JDBC的时候,忽然觉得order-by用太多了没新鲜感,我的第六感告诉我java有对线性表排序的封装,然后在eclipse里随便按了一下“.” ,哈 ...
- MYSQL根据分类分组取每组一条数据且按条件能排序的写法
之前在一个项目的开发中,有遇到要根据分类来分组获取每组一条按某个条件字段排序的数据结果,于是先自己写了一条语句: select * from `表A` GROUP BY `c`; 上面这个语句有可以根 ...
随机推荐
- mycat水平分表
和垂直分库不同,水平分表,是将那些io频繁,且数据量大的表进行水平切分. 基本的配置和垂直分库一样,我们需要改的就是我们的 schema.xml和rule.xml文件配置(server.xml不用做任 ...
- JS方法转字符串
今天接手的代码比较特殊,需要动态拼接一个table,每一行<tr>都是通过转换为字符串,再拼接在一起放到tbody中的. 其中有的td标签中有a标签,需要给a标签添加点击事件,参数好多,动 ...
- 20145208 蔡野 《网络对抗》免考项目 MSF学习
20145208 蔡野 <网络对抗>免考项目 MSF Exploit模块开发 题目:MSF Exploit模块开发 摘要 本免考项目的目标是通过对msf模块的设计语言--ruby和expl ...
- 异步任务利器Celery(二)在django项目中使用Celery
Celery 4.0支持django1.8及以上的版本,低于1.8的项目使用Celery 3.1. 一个django项目的组织如下: - proj/ - manage.py - proj/ - __i ...
- 记录结果再利用的"动态规划"之背包问题
参考<挑战程序设计竞赛>p51 https://www.cnblogs.com/Ymir-TaoMee/p/9419377.html 01背包问题 问题描述:有n个重量和价值分别为wi.v ...
- 1. 元信息:Meta类 2. 基于对象查询的sql优化 3. 自定义:Group_Concat() 4. ajax前后台交互
一.元信息 ''' 1. 元信息 1. Model类可以通过元信息类设置索引和排序信息 2. 元信息是在Model类中定义一个Meta子类 class Meta: # 自定义表名 db_table = ...
- ng-model绑定的是ng-option中的什么?
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <script sr ...
- 用vim + xdebug 来追踪thinkphp的执行过程
tree命令的使用几个有实际应用的参数 -a 这是默认的 -d: 只显式目录, 不需要显式目录下的文件 -L: 列出显式的深度. 当前目录下的所有东西为第一级... 在tp下, 有多个Common但是 ...
- 【做题】Codeforces Round #429 (Div. 2) E. On the Bench——组合问题+dp
题目大意是给你n个数,求相邻两数相乘不是完全平方数的排列数. 一开始看到这题的时候,本人便想给相乘为完全平方数的数对建边,然后就写萎了... 后来通过集体智慧发现这个重要性质:对于自然数a,b,c,若 ...
- Tomcat和weblogic虚拟路径的配置
背景:上传的图片和web应用不在同个路径里,例如web应用在D盘,上传图片1.jpg在E:\upload\img目录里,这时就需要配置虚拟路径后,才能显示图片. Tomcat和WebLogic的不同配 ...