【算法】快速排序算法的改进

image-20220104200213683

把数组分成三部分,小于划分值、等于划分值和大于划分值三部分(荷兰国旗问题),然后在 < 区 和 > 区 在用快速排序。

这种方法每次都是固定了一批数,而不是像传统的快速排序算法一样每次都只是固定了划分值的位置。

public class QuickSort {
    public static void quickSort(int[] arr) {
        if (arr == null || arr.length < 2) {
            return;
        }
        quickSort(arr, 0, arr.length -1);
    }
    //对arr[l...r]排序
    private static void quickSort(int[] arr, int l, int r) {
        if (l >= r) {
            return;
        }
        /*
        if (l > r -60) {
            //在arr[l..r]插入排序
            //O(N^2) 小样本量的时候,跑的快
            return;
        }
        */
        swap(arr, l + (int)(Math.random() * (r - l + 1)), r);
        int[] p = partition(arr, l, r);
        quickSort(arr, l, p[0] - 1); // < 区
        quickSort(arr, p[1] + 1, r); // > 区
    }

    private static int[] partition(int[] arr, int l, int r) {
        int less = l - 1; //<区右边界
        int more = r;     //>区左边界
        while (l < more) { //l表示当前数的位置
            if (arr[l] < arr[r]) { //arr[r] 划分值
                swap(arr, ++less, l++);
            } else if (arr[l] > arr[r]) {
                swap(arr, --more, l);
            } else {
                l++;
            }
        }
        swap(arr, more, r);
        return new int[] {less + 1, more};
    }

    private static void swap(int[] arr, int i, int j) {
        if (i == j) {
            return;
        }
        arr[i] = arr[i] + arr[j];
        arr[j] = arr[i] - arr[j];
        arr[i] = arr[i] - arr[j];
    }

    //for test
    //生成随机数组
    public static int[] generateRandomArray(int maxSize, int maxValue) {
        int[] arr = new int[(int) ((maxSize + 1) * Math.random())]; //长度随机
        for (int i = 0; i < arr.length; i++) {
            //值随机
            arr[i] = (int) ((maxValue + 1) * Math.random()) - (int) (maxValue * Math.random());
        }
        return arr;
    }

    public static void main(String[] args) {
        int[] arr = generateRandomArray(20,100);
        System.out.println(Arrays.toString(arr));
        quickSort(arr);
        System.out.println(Arrays.toString(arr));
    }
}

扩展:在 Java 的Arrays.sort() 方法里,数据量很小时用插入排序,数据量超过一定阈值时,对基础类型排序用的是快速排序,对对象类型排序用的是归并排序。快速排序是不稳定的,归并排序是稳定的。对于基础类型来说考虑稳定性并没有意义,所以用的是快速排序。而对于对象类型来说稳定性是比较重要的。

posted @   hzyuan  阅读(137)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· DeepSeek 开源周回顾「GitHub 热点速览」
· 记一次.NET内存居高不下排查解决与启示
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· .NET10 - 预览版1新功能体验(一)

喜欢请打赏

扫描二维码打赏

支付宝打赏

点击右上角即可分享
微信分享提示