Java集合中的排序问题

简介: 有时我们需要对集合框架(包括Set、 List、 Map)中的元素进行排序,比如在Android开发中经常要用到ListView来显示一个列表,而这个列表中的数据很多时候都是使用一个ArrayList来存储的,因为ArrayList中的元素一般都是以存储顺序排序的(向List中添加元素,一般都是调用add方法将新元素添加在末尾,当然也可以调用另一个重载的add方法将元素添加到指定位置),很明显仅按添加顺序排序是不行的,所以我们有必要对ArrayList中的元素进行排序。

有时我们需要对集合框架(包括Set、 List、 Map)中的元素进行排序,比如在Android开发中经常要用到ListView来显示一个列表,而这个列表中的数据很多时候都是使用一个ArrayList来存储的,因为ArrayList中的元素一般都是以存储顺序排序的(向List中添加元素,一般都是调用add方法将新元素添加在末尾,当然也可以调用另一个重载的add方法将元素添加到指定位置),很明显仅按添加顺序排序是不行的,所以我们有必要对ArrayList中的元素进行排序。

当然,我们也可以使用TreeSet这个类,这个类会将添加到该集合中的元素进行排序。

但在这之前,我们还有一点必须要了解的内容就是,要对集合中的元素进行排序,就需要涉及到两个对象比较的问题,即要根据对象的什么属性进行排序,凭什么你排在前面我就得排在后面。

所以为了进行比较,集合中的对象类型必须要实现Comparable接口,并同时实现该接口中的唯一一个方法compareTo方法。然后工具类Collections调用其静态方法sort来对传入的List进行排序。

集合排序的第一种情况:

我们看一下Collections类中关于sort方法的API文档说明:

public static <T extends Comparable<? super T>> void sort(List<T> list)

该方法要说明的就是要调用 Collections sort() 方法,则必须让集合中的元素实现 Comparable 接口。

不过该方法声明看起来是不是很复杂啊。我们可以慢慢分析一下。访问权限是public类型的,属于类方法(static),没有返回值(void),方法名称是sort,方法参数是List<T>类型的。我们就从这个参数类型开始分析,要传入的对象类型是List没什么好说的,根据泛型,我们知道这个List集合中存放的是T类型的,T类型,我们有学过吗?这就是让整个方法的声明变得复杂的原因所在。我们看一下返回值类型void关键字前面尖括号中的内容“T extends Comparable<? super T>”,这个内容就表示我们添加到集合List中的T类型的对象必须要实现Comparable接口(即T表示Comparable接口的子类),紧跟在Comparable后面的尖括号中的内容表示:添加到List集合中的对象可以是T类型的,也可以是T的子类型的。

这里或许还有点让你感觉到奇怪的地方就是,Comparable明明是一个接口,T是该接口的子类,关键字应该用“implements”而不应该用“extends”,是吧。实际上,这里的“extends”关键字相当于被重载了(关键字被重载可以这样说吧!),既可以表示继承也可以表示实现。(这一点,先不说了,反正你知道,T表示Comparable的子类就可以了,而Comparable接口是要排序的对象类型必须要实现的接口就行了。)

import java.util.*;
class TestSort{
public static void main(String[] args){
ArrayList<String> strList = new ArrayList<String>();
strList.add("zhangsan");
strList.add("lisi");
strList.add("wangwu");
strList.add("maer");
System.out.println(strList);
Collections.sort(strList);
System.out.println(strList);
}
}

这个例子的最终输出结果如下:

[zhangsan, lisi, wangwu, maer]
[lisi, maer, wangwu, zhangsan]

对于第一个结果没有排序是没有疑问的,按 ArrayList 中的添加顺序打印出来;关键看第二个,很明显这里的结果是经过排序的,这与我们前面介绍的,要调用 Collections sort 方法必须实现 Comparable 接口矛盾了吗,其实不矛盾,大家可以查看 API ,你会发现其实 String 是实现了 Comparable< String> 接口的,所以会对结果进行排序。当然这里举这个例子是偷个懒,你也可以让自己定义的类实现Comparable 接口并实现里面的 compareTo 方法来实现集合的排序,关于此,这里不再介绍。

下面我们介绍关于集合中排序的第二种情况:就是我想将集合中的元素按不同的规则来排序。既可以按对象的名称的自然顺序排序,也可按对象的其他属性排序,那这时应该怎么实现呢?

如果用第一种情况的方法的话,让我们的类实现Comparable接口,并实现里面的compareTo方法,但这样我们只能得到一种排序方式。

我们可以继续查看一下CollectionsAPI文档,我们会发现在CollectionsAPI文档中,还存在一个重载的sort方法,其完整的方法声明如下:

public static <T> void sort(List<T> list, Comparator<? super T> c)

我们关键看下它的参数,共有两个参数,一个 List<T> 对象,一个 Comparator <? super T>对象。这里我们主要对 Comparator 类型陌生,我们看一下 Comparator API ,其声明如下:

public interface Comparator<T>{
int compare(T o1, T o2);
}

注:该接口中还有一个 equals 方法,因为与这里不相关,就没有列出。

关于该类的具体介绍可以查看API,这里只要明白,要实现多种方式排序就要按排序方式多次实现该接口,并实现其中的compare方法。关于此就不再举例了。

相关文章
|
1月前
|
安全 Java 容器
【Java集合类面试二十七】、谈谈CopyOnWriteArrayList的原理
CopyOnWriteArrayList是一种线程安全的ArrayList,通过在写操作时复制新数组来保证线程安全,适用于读多写少的场景,但可能因内存占用和无法保证实时性而有性能问题。
|
1月前
|
存储 安全 Java
【Java集合类面试二十五】、有哪些线程安全的List?
线程安全的List包括Vector、Collections.SynchronizedList和CopyOnWriteArrayList,其中CopyOnWriteArrayList通过复制底层数组实现写操作,提供了最优的线程安全性能。
|
1月前
|
Java
【Java集合类面试二十八】、说一说TreeSet和HashSet的区别
HashSet基于哈希表实现,无序且可以有一个null元素;TreeSet基于红黑树实现,支持排序,不允许null元素。
|
1月前
|
Java
【Java集合类面试二十六】、介绍一下ArrayList的数据结构?
ArrayList是基于可动态扩展的数组实现的,支持快速随机访问,但在插入和删除操作时可能需要数组复制而性能较差。
|
1月前
|
存储 Java 索引
【Java集合类面试二十四】、ArrayList和LinkedList有什么区别?
ArrayList基于动态数组实现,支持快速随机访问;LinkedList基于双向链表实现,插入和删除操作更高效,但占用更多内存。
|
6天前
|
安全 Java API
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
String常量池、String、StringBuffer、Stringbuilder有什么区别、List与Set的区别、ArrayList和LinkedList的区别、HashMap底层原理、ConcurrentHashMap、HashMap和Hashtable的区别、泛型擦除、ABA问题、IO多路复用、BIO、NIO、O、异常处理机制、反射
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
|
21天前
|
Java
用JAVA架建List集合为树形结构的代码方法
这段代码定义了一个表示树形结构的 `Node` 类和一个用于构建树形结构的 `TreeController`。`Node` 类包含基本属性如 `id`、`pid`、`name` 和 `type`,以及子节点列表 `children`。`TreeController` 包含初始化节点列表并将其转换为树形结构的方法。通过过滤和分组操作实现树形结构的构建。详情可见:[代码示例链接1](http://www.zidongmutanji.com/zsjx/43551.html),[代码效果参考链接2](https://www.257342.com/sitemap/post.html)。
28 5
|
21天前
|
存储 Java 程序员
Java中的集合框架:从入门到精通
【8月更文挑战第30天】在Java的世界里,集合框架是一块基石,它不仅承载着数据的存储和操作,还体现了面向对象编程的精髓。本篇文章将带你遨游Java集合框架的海洋,从基础概念到高级应用,一步步揭示它的奥秘。你将学会如何选择合适的集合类型,掌握集合的遍历技巧,以及理解集合框架背后的设计哲学。让我们一起探索这个强大工具,解锁数据结构的新视角。
|
22天前
|
存储 算法 Java
Java中的集合框架深度解析云上守护:云计算与网络安全的协同进化
【8月更文挑战第29天】在Java的世界中,集合框架是数据结构的代言人。它不仅让数据存储变得优雅而高效,还为程序员提供了一套丰富的工具箱。本文将带你深入理解集合框架的设计哲学,探索其背后的原理,并分享一些实用的使用技巧。无论你是初学者还是资深开发者,这篇文章都将为你打开一扇通往高效编程的大门。
|
23天前
|
存储 Java
Java中ArrayList 元素的排序
本文提供了Java中根据`ArrayList`元素的某个属性进行排序的示例代码,包括实现`Comparable`接口和重载`compareTo`方法,然后使用`Collections.sort`方法进行排序。