快速排序 quick sort

快速排序

  • 时间复杂度:平均、最好为O(nlogn),最坏为O(n^2)
  • 空间复杂度:O(nlogn)
  • 稳定性:不稳定

算法分析:

  • 快速排序使用了分治思想来把一个数组分成两个子数组(以数组为例)
  • 快速排序从数组中选择一个元素为基准(pivot)
  • 把比基准大的元素放到基准右边
  • 把比基准小的放到基准左边
  • 这样以基准为中心,就把原来的数组分成了两个子序列
  • 基准左边大于基准的 和 基准右边小于基准的交换位置(双指针最快)
  • 基准归位
  • 然后把基准分开的两个子序列递归的重复上述步骤
  • 最后整个数组排序完毕

动画演示:(动画来自五分钟学算法,侵删)

quickSort

C代码
void quickSort(int arr[],int left,int right) {
    if (left > right) { return ; }
    int i,j,tmp,pivot;
    i = left;
    j = right;
    pivot = arr[left];
    while (i != j) {
        while (arr[j] >= pivot && i < j) {
            j--;
        }
        while (arr[i] <= pivot && i < j) {
            i++;
        }
        if (i < j) {
            tmp = arr[i];
            arr[i] = arr[j];
            arr[j] = tmp;
        }
    }
    //基准归位
    arr[left] = arr[i];
    arr[i] = pivot;
    //递归
    quickSort(arr, left, i-1);
    quickSort(arr, i+1, right);
}
OC代码
- (void)quickSort:(NSMutableArray *)arr left:(int)left right:(int)right {
    if (left > right) {
        return;
    }
    NSNumber *tmp = nil;
    int i = left;
    int j = right;
    NSNumber *pivot = arr[left];
    while (i != j) {
        while ([arr[j] intValue] >= pivot.intValue && i < j) {
            j--;
        }
        while ([arr[i] intValue] <= [pivot intValue] && i < j) {
            i++;
        }
        if (i < j) {
            tmp = arr[i];
            arr[i] = arr[j];
            arr[j] = tmp;
        }
    }
    arr[left] = arr[i];
    arr[i] = pivot;
    
    [self quickSort:arr left:left right:i-1];
    [self quickSort:arr left:i+1 right:right];
}
Swift代码
func quickSort(arr: inout [Int],left:Int,right:Int) -> Void {
    if left > right { return }
    var i = left,j = right
    var pivot = arr[left],tmp = 0
    while i != j {
        while (arr[j] >= pivot && i < j) {
            j -= 1
        }
        while (arr[i] <= pivot && i < j) {
            i += 1
        }
        if i < j {
            tmp = arr[i];
            arr[i] = arr[j]
            arr[j] = tmp
        }
    }
    arr[left] = arr[i]
    arr[i] = pivot
    
    quickSort(arr: &arr, left: left, right: i-1)
    quickSort(arr: &arr, left: i+1, right: right)
}

以下代码实现来源于网络,未验证

js代码实现
function quick_sort(arr,from,to){
    var i = from; 
    var j = to; 
    var key = arr[from]; //标准值
    if(from >= to){ //如果数组只有一个元素
       return;
    }
    while(i < j){
        while(arr[j] > key && i < j){ //从右边向左找第一个比key小的数,找到或者两个哨兵相碰,跳出循环
            j--;
        }
        while(arr[i] <= key && i < j){  //从左边向右找第一个比key大的数,找到或者两个哨兵相碰,跳出循环,这里的=号保证在本轮循环结束前,key的位置不变,否则的话跳出循环,交换i和from的位置的时候,from位置的上元素有可能不是key
            i++;
        }
        if(i < j){ //交换两个元素的位置
            var temp = arr[i];
            arr[i] = arr[j];
            arr[j] = temp;
        }
    }
    arr[from] = arr[i] //
    arr[i] = key;
    quick_sort(arr,from,i-1);
    quick_sort(arr,i+1,to);
}
Java实现
public static void sort(int[] a, int low, int high) {
    if(low>=high)
        return;
    int i = low;
    int j = high;
    int key = a[i];
    while (i < j) {
        while (i < j && a[j] >= key)
            j--;
        a[i++] = a[j];
    while (i < j && a[i] <= key)
            i++;
        a[j--] = a[i];
    }
    a[i] = key;
    sort(a,low,i-1);
    sort(a,i+1,high);
}
c++实现
 void QuickSort(int array[], int start, int last) {
    int i = start;
    int j = last;
    int temp = array[i];
    if (i < j)
    {
        while (i < j) {
            while (i < j &&  array[j]>=temp )
                j--;
            if (i < j) {
                 array[i] = array[j];
                 i++;
             }
            while (i < j && temp > array[i])
                 i++;
             if (i < j) {
                 array[j] = array[i];
                 j--;
             }
         }
         //把基准数放到i位置
         array[i] = temp;
         //递归方法
         QuickSort(array, start, i - 1);
         QuickSort(array, i + 1, last);
     }
 }
python实现
def quick_sort(array, left, right):
    if left >= right:
        return
    low = left
    high = right
    key = array[low]
    while left < right:
        while left < right and array[right] > key:
            right -= 1
        array[left] = array[right]
        while left < right and array[left] <= key:
            left += 1
        array[right] = array[left]
    array[right] = key
    quick_sort(array, low, left - 1)
    quick_sort(array, left + 1, high)
go实现
func quickSort(arr []int){
    arrLen := len(arr)
    if arrLen <= 1{
        return
    }

    randNum := getRandNum(arrLen -1)
    arr[randNum], arr[0] = arr[0],arr[randNum]
    mid := arr[0]//取出用于比较的元素

    head, tail := 0, arrLen -1

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

推荐阅读更多精彩内容