Collections.sort()的分析

本文深入探讨了Java中Collections.sort方法的工作原理,详细解释了为何需要将List转换为数组进行排序,以及排序后如何将结果重新填入原始List。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

首先我们得说明在Collections里面有两个排序方法
 public static <T> void sort(List<T> list, Comparator<? super T> c)
 public static <T extends Comparable<? super T>> void sort(List<T> list) 
 很清晰,第二个方法不需要传入比较器,但是传入的待排序的参数list必须实现Comparable接口。
 我们目前就只看第二个sort。
Collections.java
    public static <T extends Comparable<? super T>> void sort(List<T> list) {
        Object[] a = list.toArray();
        Arrays.sort(a);       //标识1
        ListIterator<T> i = list.listIterator();
        for (int j=0; j<a.length; j++) {
            i.next();
            i.set((T)a[j]);
        }
    }
从上面我们可以清晰的看到,首先将要排序的list转化成数组。
再对数组使用Arrays.sort进行排序。
    Arrays.java
    public static void sort(Object[] a) {
        if (LegacyMergeSort.userRequested)
            legacyMergeSort(a);
        else
            ComparableTimSort.sort(a);
    }
Arrays的排序,我们不必深究。
总而言之,代码运行标志1后,数组a已经是有序的了。

为什么要将list转化成数组呢?
因为java内部的排序就只有针对数组的。
那么后面的是干什么呢?(list.listIterator()部分及以后)
举个例子
原list             1 5 8 3 4 2 6
排序后的数组 1 2 3 4 5 6 8
ListIterator<T> i = list.listIterator();
上面的代码取得了原list的迭代器,然后不断的将排序后的数组替换进原list(这个过程就是不断的next移动标尺,放入,再移动..)。


那么改成这样如何呢?
    @SuppressWarnings("unchecked")
    public static <T extends Comparable<? super T>> List<T> sort(List<T> list) {
        Object[] a = list.toArray();
        Arrays.sort(a);
        return (List<T>) Arrays.asList(a);
    }
好理解了一下,不过额外生成了一个list。
真的是生成了一个list么?
    Arrays.java
    public static <T> List<T> asList(T... a) {
        return new ArrayList<>(a);
    }
返回的是ArrayList!!!
换言之
        List<Integer> a=new LinkedList<Integer>();
        a.add(4);
        a.add(7);
        a.add(5);
        a.add(6);
        a.add(8);
        a.add(2);

        LinkedList<Integer> list=(LinkedList<Integer>) sort(a);
这样的代码就会出错!!!
### Java `Collections.sort` 方法详解 #### 1. 基本概念 `Collections.sort()` 是 Java 集合框架中的一个重要工具方法,用于对集合对象(如 `List`)进行排序。它有两种重载形式: - **无参数的 `sort(List<T> list)`**:适用于实现了 `Comparable` 接口的对象列表。 - **带自定义比较器的 `sort(List<T> list, Comparator<? super T> c)`**:允许通过传入 `Comparator` 实现灵活的排序逻辑。 此方法内部实际上调用了 `Arrays.sort()` 来完成具体的排序工作[^3]。 --- #### 2. 使用示例 ##### 示例 1:基本排序(自然顺序) 当列表中的元素实现了 `Comparable` 接口时,可以直接使用无参数的 `Collections.sort()` 方法。 ```java import java.util.ArrayList; import java.util.Collections; import java.util.List; public class SortExample { public static void main(String[] args) { List<Integer> numbers = new ArrayList<>(); numbers.add(5); numbers.add(2); numbers.add(8); numbers.add(1); Collections.sort(numbers); // 自然顺序排序 System.out.println("Sorted Numbers: " + numbers); } } ``` 上述代码会按照整数的小从小到排列输出结果为 `[1, 2, 5, 8]`[^1]。 --- ##### 示例 2:自定义排序(使用 `Comparator`) 对于未实现 `Comparable` 的类或者需要特殊排序规则的情况,可以提供一个 `Comparator` 对象来控制排序行为。 ```java import java.util.*; class Person implements Comparable<Person> { String name; int age; Person(String name, int age) { this.name = name; this.age = age; } @Override public int compareTo(Person other) { return Integer.compare(this.age, other.age); // 按年龄升序排序 } @Override public String toString() { return name + "(" + age + ")"; } } public class CustomSortExample { public static void main(String[] args) { List<Person> people = Arrays.asList( new Person("Alice", 30), new Person("Bob", 25), new Person("Charlie", 35) ); // 默认按年龄排序 Collections.sort(people); System.out.println("Default Sorted People: " + people); // 按姓名降序排序 Collections.sort(people, (p1, p2) -> p2.name.compareTo(p1.name)); System.out.println("Name Descending Sorted People: " + people); } } ``` 这段代码展示了如何先基于默认的 `compareTo` 方法按年龄排序,再利用 Lambda 表达式的 `Comparator` 按名字倒序排列[^2]。 --- #### 3. 底层原理分析 `Collections.sort()` 调用的核心在于其最终依赖的是 `Arrays.sort()` 函数。具体来说,在现代 JDK 中(Java 7 及以上),主要采用了一种名为 TimSort 的高效混合排序算法[^5]。这种算法结合了归并排序和插入排序的优点,具有良好的性能表现,尤其适合处理部分已有序的数据集。 以下是简化版的流程描述: 1. 将目标 `List` 转化成数组; 2. 利用 `Arrays.sort()` 执行实际排序操作; 3. 把排好序的结果重新填充回原 `List` 结构中。 --- #### 4. 注意事项 - 若尝试对不支持 `Comparable` 或者没有提供适当 `Comparator` 的对象执行排序,则会发生编译错误或运行期异常。 - 当涉及复杂数据结构时,请确保遵循 `equals` 和 `hashCode` 协议以维持一致性[^4]。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值