最近在刷一道算法题 《字符串重新排序》时,发现自己有思路但是写代码的时候就无从下手了 而且看了答案之后还没看懂 关键就是基础不好 对于排序没有理解(虽然我学过常用的排序算法 但是都是理念 实践少)
从实践和原理出发 重点是从实践出发 探讨如何使用 sort()方法 完成复杂的排序
能掌握到的知识
目录
sort()方法有很多种
public void sort(java.util.Comparator super E> c )
通过返回 1 0 -1 等三个数 比较列表中对象属性值的方法 实现的排序
如下所示
userList.sort((o1,o2) ->{
if (o1.getAge() > o2.getAge()) {
//降序
return -1;
} else if (o1.getAge() < o2.getAge()) {
return 1;
} else {
return 0;
}
});
那1 0 -1 分别代表什么呢 怎么比较能够实现升序降序呢
PS:这里并不一定是1 和-1 只要是负整数和正整数就行了 只不过我们习惯上用1和-1表示
放两个结论
所以
降序你可以自己总结 下面我们实战演示一下
给出测试数据
@Data
@NoArgsConstructor
@AllArgsConstructor
public class User {
private int age;
private int high;
private String name;
}
例子
public static void main(String[] args) {
List<User> userList = new ArrayList<>();
userList.add(new User(18, 165,"张三1"));
userList.add(new User(16, 177, "张三11"));
userList.add(new User(25, 189, "张三4"));
userList.add(new User(25, 167,"张三3"));
userList.add(new User(16, 155, "张三8"));
userList.sort((o1,o2) ->{
if (o1.getAge() > o2.getAge()) {
//降序
return -1;
} else if (o1.getAge() < o2.getAge()) {
// 降序
return 0;
} else {
return 0;
}
});
// userList.sort((o1,o2) ->{
// if (o1.getAge() < o2.getAge()) {
// //升序
// return -1;
// } else if (o1.getAge() > o2.getAge()) {
// return 0;
// } else {
// return 0;
// }
//
// });
System.out.println("userList = " + userList);
}
很多时候我们可能并不是只按一个属性进行排序 比如以下要求
- 先按照年龄升序
- 再按照身高降序
public static void main(String[] args) {
List<User> userList = new ArrayList<>();
userList.add(new User(18, 165,"张三1"));
userList.add(new User(16, 177, "张三11"));
userList.add(new User(25, 189, "张三4"));
userList.add(new User(25, 167,"张三3"));
userList.add(new User(16, 155, "张三8"));
userList.sort(((o1, o2) -> {
//按照年龄升序
if (o1.getAge() > o2.getAge()) {
//后面的大于前面的数 数越来越大 即升序 所以不需要交换顺序
return 1;
} else if (o1.getAge() < o2.getAge()) {
// 后面的小于前面的数 数越来越小 即降序 但是我们想要升序 所以需要交换顺序 返回-1
return -1;
} else {
//相等时 按照身高降序
if (o1.getHigh() > o2.getHigh()) {
// 后面的大于前面 即数越来越大 升序 但是我们想要降序 所以返回-1
return -1;
} else if (o1.getHigh() < o2.getHigh()) {
return 1;
} else {
return 0;
}
}
}));
System.out.println("userList = " + userList);
}
题目描述
有一串单词组成的英文字符串 需要下面要求进行排序
- 统计每个单词出现的次数,并按次数降序排列;
- 次数相同时,按单词长度升序排列;
3)次数和单词长度均相同时,按字典序升序排列
举例
示例1
输入:
This is an apple
输出:
an is This apple
示例2
输入
Wisdom in the mind is better than money in the hand
输出
in in the the is hand mind than money Wisdom better
代码
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
String originString = scanner.nextLine();
//1. 获取全部
List<String> list = new ArrayList<>(Arrays.asList(originString.split(" ")));
//2. 统计每个单词出现的次数
Map<String, Long> countMap = statistic(list);
// 3. 排序
List<Map.Entry<String, Long>> collect = countMap.entrySet().stream()
.sorted(((o1, o2) -> {
if (o2.getValue() < o1.getValue()) {
// 降序
return -1;
} else if (o1.getValue().equals(o2.getValue())) {
if (o2.getKey().length() > o1.getKey().length()) {
// 升序 o2是后面那个
return -1;
} else if (o1.getKey().length() == o2.getKey().length()) {
return o1.getKey().compareTo(o2.getKey());
}
}
return 1;
})).collect(Collectors.toList());
//4. 根据key 和value次数输出结果
StringBuilder builder = new StringBuilder();
for (Map.Entry<String, Long> entry : collect) {
Long value = entry.getValue();
String key = entry.getKey();
for (int i = 0; i < value.intValue(); i++) {
builder.append(key).append(" ");
}
}
System.out.println(builder.substring(0, builder.length() - 1).toString());
}
private static Map<String, Long> statistic(List<String> list) {
Map<String, Long> collect = list.stream()
.collect(Collectors.groupingBy(Function.identity(), Collectors.counting()));
return collect;
}
我们主要关注第3点 排序那里的代码
参考:
CSDN博主「秃秃爱健身」的原创文章:https://blog.csdn.net/Saintmm/article/details/125218362