美文网首页
(单路,双路,三路)快速排序讲解及Java实现

(单路,双路,三路)快速排序讲解及Java实现

作者: TheTempest | 来源:发表于2020-01-06 13:24 被阅读0次

快速排序(简称快排):在待排序数组中确定一个基准值(pivot),一次排序后将所有小于基准值的数移动至基准值左侧,大于基准值的数据移动至基准值右侧,这样基准值所在的位置就是最终排序后其应在位置。根据分治、递归的思想,对左右两侧数据递归上面的操作,直至区间缩小为1,所有的数据就都有序了。

基准值的选取方法是很关键的(比如三元选取,随机选取等),但是并属于该篇博客的讲解范围,所以下面为了解释方便,基准值暂定的比较随意,大家谅解 ~

单路快排

核心思想:基准值定为最右边,i和j从最左边开始,如果j小于基准值,则i和j交换位置,并且i++,j++。否则i保持不动,j++。最终当j移动到基准值所在位置后,基准值与i交换位置。

下图中说明了一次排序的详细流程: 单路快排图示(*注:该图来自极客时间) 下面按照上述的规则列出一组数据整个交换流程: 单路排序数据交换流程

名词解释 -- 稳定性:经过某种排序算法之后,如果相同值的数据,前后顺序没有发生改变,我们就把这种算法叫做稳定的排序算法。

猜想一:基准值从最左边开始是否可以? 猜想一

得出结论:基准值必需是j移动的结尾,因为最终需要一次基准值和i的交换位置

代码实现:

    public static void quickSortSingle(int[] arr, int left, int right) {
        if (left > right)
            return;

        int pivot = arr[right];
        int i = 0, j = 0;
        while (j < right) {
            while (j < right && arr[j] <= pivot) {//如果"哨兵"j小于基准值,则"哨兵"i与"哨兵"j交换位置
                swap(arr, i, j);
                i++;
                j++;
            }
            j++;
        }
        //此时"哨兵"j移动到最右侧,基准值与哨兵"i"所在位置的值进行交换
        swap(arr, i, right);

        quickSortSingle(arr, left, i - 1);
        quickSortSingle(arr, i + 1, right);
    }

    private static void swap(int[] arr, int i, int j) {
        int tmp = arr[i];
        arr[i] = arr[j];
        arr[j] = tmp;
    }

双路快排

核心思想:基准值在最左边,“哨兵”i在最左边,“哨兵”j在最右边,从右边注意要从右边开始,下面会有说明)先开始(j--),如果“哨兵”j所在的数据小于基准值则停止;“哨兵”i开始(i++),如果“哨兵”i所在的数据大于基准值则停止,i与j交换位置;如果i和j相遇,则基准值与i或j(因为两者现在一致)交换位置。

下面按照上述的规则列出一组数据整个交换流程: 双路排序数据交换流程 猜想二:如果先从左边找会出现什么问题? 猜想二
得出结论:因为如果先从右边开始会停留在比基准值大的数上,这时交换基准值的位置就会出现问题,所以开始执行的方向必需是基准值的反方向。

代码实现:

    public static void quickSortDual(int[] arr, int left, int right) {
        if (left > right)
            return;

        int pivot = arr[left];//基准值
        int i = left;//左侧"哨兵"
        int j = right;//右侧"哨兵"
        while (i != j) {//注意:要从基准值所在侧的另外一侧开始
            while (arr[j] >= pivot && i < j)//如果右侧出现了比基准值小的元素,则"哨兵"j停留
                j--;
            while (arr[i] <= pivot && i < j)//如果左侧出现了比基准值小的元素,则"哨兵"i停留
                i++;
            if (i < j) {//如果"哨兵"i与j没有相遇则交换其所在位置的数据
                int tmp = arr[i];
                arr[i] = arr[j];
                arr[j] = tmp;
            }
        }
        //此时"哨兵"i与j相遇,交换基准值与该相遇点的数据
        arr[left] = arr[i];
        arr[i] = pivot;

        quickSortDual(arr, left, i - 1);//递归的处理左侧数据
        quickSortDual(arr, i + 1, right);//递归的处理右侧数据
    }

三路快排

核心思想:基准值在最左边,“哨兵”i在基准值右侧加1的位置,“哨兵”j在最右边,从基准值右侧加1的位置开始往后遍历,如果遍历到的当前值小于基准值,则当前值与左侧"哨兵"交换位置,左侧"哨兵"进一,反之,则当前值与右侧"哨兵"交换位置,左侧"哨兵"退一。

下面按照上述的规则列出一组数据整个交换流程(虽然该数据的排序过程并没有丢失稳定性,但是大家不要认为三路快排是个稳定排序算法): 三路排序数据交换流程

代码实现:

    public static void quickSortThreeWay(int[] arr, int left, int right) {
        if (left > right)
            return;

        int pivot = arr[left];//基准值
        int i = left;//左侧"哨兵"
        int j = right + 1;//右侧"哨兵"
        int index = left + 1;
        while (index < j) {
            if (arr[index] < pivot) {//如果当前值小于基准值,则当前值与左侧"哨兵"交换位置,左侧"哨兵"进一
                swap(arr, index, i + 1);
                i++;
                index++;
            } else if (arr[index] > pivot) {//如果当前值大于基准值,则当前值与右侧"哨兵"交换位置,左侧"哨兵"退一
                swap(arr, index, j - 1);
                j--;
            } else {
                index++;
            }
        }

        swap(arr, left, i);

        quickSortSingle(arr, left, i - 1);
        quickSortSingle(arr, j, right);
    }

    private static void swap(int[] arr, int i, int j) {
        int tmp = arr[i];
        arr[i] = arr[j];
        arr[j] = tmp;
    }

相关文章

  • (单路,双路,三路)快速排序讲解及Java实现

    快速排序(简称快排):在待排序数组中确定一个基准值(pivot),一次排序后将所有小于基准值的数移动至基准值左侧,...

  • OC数据结构&算法

    更多整理资料尽在?一平米小站 目录 选择排序 冒泡排序 插入排序 快速排序 双路快速排序 三路快速排序 堆排序 选...

  • 算法:排序和搜索

    75 颜色分类快速排序:基于重复元素过多的问题,有双路排序和三路排序算法。双路排序即最常用的写法:参考:https...

  • 排序

    插入排序 选择排序 冒泡排序 归并排序 快速排序* 三路快速排序

  • 排序算法总结

    选择排序法 插入排序法 冒泡排序法 归并排序法 自顶向下 自底向上 快速排序法 单路快速排序法 双路快速排序法 三...

  • golang 写个快速排序

    快速排序是大多数语言内置 sort 函数的默认实现方式,简单可分为两路排序和三路排序,我在相关资料中,发现两路排序...

  • python快速排序

    探究快速排序及三路快速排序的应用场景在对近乎有序或相同元素很多的列表进行排序时,如果不进行调优,快速排序会接近o(...

  • 面试知识点

    排序冒泡排序快速排序选择排序插入排序二路归并 查找二分查找 排序和查找的java实现 java语言Java字符串字...

  • 三路快速排序

    对于快速排序,如果排序的数组中有很多的重复数如10000个【1,9】的数就有很多重复数,由于快速排序的判定问题,会...

  • 快速排序(java实现)

    下面这篇文章是对快速排序讲解的比较清晰明了的,可以参考学习。 快速排序(java实现)

网友评论

      本文标题:(单路,双路,三路)快速排序讲解及Java实现

      本文链接:https://www.haomeiwen.com/subject/kathactx.html