排序算法之--快速排序

之所以叫快速排序,是因为快排在实际应用中是表现最好的排序算法。

快速排序采用分治策略对数据进行排序,什么是分治策略呢?简单地说就是“分而治之,各个击破”。啥意思呢?且听我慢慢道来:

假设现在给你一个数组int[] arr = {6, 2, 4, 9, 3, 10},选择一个基准数(这个可以随便选,为了方便一般选择第一个元素为基准数)pivot = arr[0] = 6;将大于基准数pivot的元素放在pivot右侧,小于pivot的元素放在pivot左侧;这样基准值就处在它应该处在的位置了;然后对基准值左右两侧的子数组再选取基准值进行一轮如上的操作,这样不断的将数组一份为二,每一次都“安顿”好一个数值(基准值),最后一定能够将整个数组排序(“安顿”)好。

现在把上面的叙述抽象成步骤:

  1. 从数组中挑选一个基准值,并设置i = 0, j = a.length - 1;
  2. 先从右往左寻找比基准值小的值,再从左往右寻找比基准值大的值,交换它们;重复此步骤,直到 i 和 j 相遇
  3. 将基准值和a[i]交换以完成归位
  4. 递归地把"基准值前面的子数组"和"基准值后面的子数组"进行步骤1,2,3

针对上面的步骤有几点说明:

  • 基准值一般选择当前数组的第一个元素,这样做是为了代码实现方便,更好理解
  • 对于和基准值相等的数值放在左边或右边都可以,同时这也是快速排序不稳定的原因
  • 经过步骤2后我们就把基准值“安顿”好了
  • 步骤3所谓递归,其实就是不断地把基准值“安顿”好

说的再好都不如图来的畅快:

其实,快速排序算法分为两个部分:分段(Partition)和递归(Recursive)。代码实现如下:

C++代码实现

#include <iostream>
using namespace std;

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

void QuickSortRec(int arr[], int i, int j)
{
    if(i > j)
        return;

    // 以最左侧元素作为pivot基准元素
    int left = i;
    int right = j;
    int pivot = arr[left];
    
    while(left < right)
    {
        // 从right开始往左寻找 < pivot元素
        while(right > left)
        {
            if(arr[right] < pivot)
            {
                break;
            }
            right--;
        }
        
        // 从left开始往右寻找 > pivot元素
        while(left < right)
        {
            if(arr[left] > pivot)
            {
                break;
            }
            left++;
        }
        
        // left找到了大于pivot的,right找到了小于pivot的,交换
        swap(arr, left, right);
    }
   
    // 到此处,左右游标相遇,交换pivot和arr[left]
    swap(arr, i, left);
    
    // 递归pivot左右序列
    QuickSortRec(arr, i, left - 1);
    QuickSortRec(arr, left + 1, j);
    
}

int main()
{
    int arr[] = {2, 12, 23, 6, 89, 1, 2, 3, 7, 6, 5};
    int len = sizeof(arr)/sizeof(arr[0]);
    QuickSort(arr, 0, len - 1);
    
    for(auto& v : arr)
    {
        cout<<v<<" ";
    }
    
    return 0;
}

Java代码实现

public class TestSort {
    public static void main(String[] args) {
        int[] arr = {12333, 34, 2, 1, 3, 5, 64, 2, 9, 12, 4, 8};
        quickSort(arr, 0, arr.length - 1);
        for (int i : arr) {
            System.out.println(i);
        }
    }

    /*
         arr:   待排序数组
         left:  排序起始位置
         right: 排序终止位置
     */
    private static void quickSort(int[] arr, int left, int right) {
        // ##################################### 分段开始 ####################################
        if (left > right || arr == null || arr.length == 0) {
            return;
        }
        int pivot = arr[left];//以左边界的值为基准
        int i = left;
        int j = right;//使用局部变量接收参数

        while (i != j) { //当i = j时表示已经排序好(即基准点左侧都小于基准点,基准点右侧都大于基准点)
            while (arr[j] >= pivot && i < j) { //从右往左找小于基准点的数
                j--;
            }
            while (arr[i] <= pivot && i < j) { //从左往右找大于基准点的数
                i++;
            }
            if (i < j) { //寻找到的大于/小于基准点的下标还没有相遇,交换两个的位置
                swap(arr, i, j);
            }
        }
        //代码到这,即表示 i, j 已经相遇了

        //将基准数归位
        swap(arr, left, i);
        // ##################################### 分段结束 ####################################

        // ##################################### 递归开始 ####################################
        quickSort(arr, left, i - 1);//递归对左侧子数组排序
        quickSort(arr, i + 1, right);//递归对右侧子数组排序
        // ##################################### 递归结束 ####################################
    }
    //交换数组arr中下标from和to的元素
    private static void swap(int[] arr, int from, int to) {
        if (from < 0 || from >= arr.length || to < 0 || to >= arr.length) return;
        int tmp = arr[from];
        arr[from] = arr[to];
        arr[to] = tmp;
    }

}

快排的思想可以总结为:冒泡 + 二分 + 递归分治。

复杂度、稳定性

时间复杂度:O(nlogn)。
空间复杂度:O(nlogn)。
稳定性:不稳定。

为什么一定要从右边开始呢

如以上的程序中,我们在以基准数交换数时,是先从右边开始(j):

while (arr[j] >= pivot && i < j) { //从右往左找小于基准点的数
    j--;
}
while (arr[i] <= pivot && i < j) { //从左往右找大于基准点的数
    i++;
}

可不可以从左边开始呢:

while (arr[i] <= pivot && i < j) { //从左往右找大于基准点的数
    i++;
}
while (arr[j] >= pivot && i < j) { //从右往左找小于基准点的数
    j--;
}

实测发现,这样是不行的。

想象数组{6, 1, 2, 7, 9},以6为基准数进行快速排序。

  • 从左侧开始探索

先从左边开始探索,i将落在7的位置;再从右侧探索,j也将落在7的位置。
此时交换基准数和i指向的数7,数组变成{7, 1, 2, 6, 9}。
考察这个数组,基准数6左边的数并不全是小于基准数的,第一个位置上的7就是比6大的。

  • 从右侧开始探索

先从右边开始探索,j将落在2上;再从左侧探索,i也将落在i上。
此时交换基准数和i指向的数2,数组变成{2, 1, 6, 7, 9}。
考察这个数组,基准数6左侧的都比6小,右侧的都比6大。

形而上一点,如果选取最左侧的数arr[left]作为基准数,从最右侧开始探索可以保证当i,j相遇时,i对应的数是小于基准数的,此时交换基准数和i对应的数可保证基准数左侧的数都小于基准数。而如果从左侧开始探索,则当i,j相遇时i对应的数是大于基准数的,此时交换基准数和i对应的数就无法满足基准数左侧的值都小于基准数。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容