算法笔记p142快速排序

本文详细介绍了快速排序算法的划分过程,包括传统的选择第一个元素作为枢轴和改进的随机选择枢轴方法,以及如何通过双指针技巧和递归实现快速排序。特别强调了在元素接近有序时使用随机划分来避免最坏时间复杂度的方法。
摘要由CSDN通过智能技术生成

快速排序

快速排序是排序算法中平均时间复杂度为O(nlogn)的一种算法。

划分

快排的实现需要解决划分的问题:对于一个序列A[1]、A[2]、……、A[n],从中选取一个枢轴(或主元),如取A[1]为枢轴,调整序列中元素的位置,使得枢轴左侧所有元素都不超过枢轴,右侧所有元素都大于枢轴。
双指针思想:

  1. 先将A[1]存放至某个临时变量temp,并令两个下标left、right分别指向序列首尾(如令left = 1、right = n)。
  2. 只要right指向的元素A[right]大于temp,就将right不断左移;当某个时候A[right] ≤ temp时,将元素A[right]挪到left指向的元素A[left]处。
  3. 只要left指向的元素A[left]不超过temp,就将left断右移;当某个时候A[left] > temp时,将元素A[left]挪到right指向的元素A[right]处。
  4. 重复②③,直达left与right相遇,把temp(也即原A[1])放到相遇的地方。

以A[left]为枢轴,写出划分区间代码:

// 对区间[left,right]进行划分
int partition(int left, int right, int A[]) {
    int temp = A[left];			// 将A[left]存放至临时变量temp
    while (left < right) {		// left == right,即枢轴位置
        while (left < right && temp < A[right]) right--;	// 反复左移right
        A[left] = A[right];		// 将A[right]挪到A[left]
        while (left < right && A[left] <= temp) left++;		// 反复右移left
        A[right] = A[left];		// 将A[left]挪到A[right]
    }
    A[left] = temp;				// 把temp放到left与right相遇的地方
    return left;				// 返回相遇的下标
}

快排

  1. 调整序列中的元素,使当前序列最左端的元素在调整后满足左侧所有元素均不超过该元素,右侧所有元素均大于该元素。
  2. 对该元素的左侧和右侧分别递归进行①的调整,直到当前调整区间的长度不超过1。
// 快速排序,left与right初值为序列首尾下标(如1与n)
void quick_sort(int left, int right, int A[]) {
    if (left < right) {                             // left == right时,区间只有一个元素
        int pivot = partition(left, right, A);      // 将[left,right]按A[left]一分为二
        quick_sort(left, pivot - 1, A);         	// 对左子区间递归进行快速排序
        quick_sort(pivot + 1, right, A);        	// 对右子区间递归进行快速排序
    }
}

随机划分的快速排序

  1. 当序列中元素接近有序时,选取A[left]作为主元,快速排序可能会达到最坏时间复杂度O(n2)。
  2. 采用随机选择主元的方法,则对任意输入数据的期望时间复杂度都能达到O(nlogn)。
  3. C语言中产生随机数据:
    1. 需要添加stdlib.h头问题与time.h头文件。
    2. 生成随机数的种子:写上srand((unsigned) time(NULL));(记住即可)。
    3. 在需要使用随机数的地方使用rand()函数。
    4. 生成[left, right]范围内的随机数:
      1. rand()函数生成[0,RAND_MAX]范围内的随机数。
      2. 用这个随机数除以RAND_MAX得到一个[0,1]范围内的浮点数。
      3. 用这个浮点数乘以(right - left),加上left。
      4. 再用round()函数(需要math.h头文件)四舍五入后,得到[left,right]范围内的随机数。
      5. 这个浮点数相当于[left,right]范围内的比例位置。
      6. 例:rand()生成的随机数经过处理得到浮点数为0.5,left = 2,right = 5,right - left = 3,3 * 0.5 = 1.5,1.5 + 2 = 3.5,round(3.5) = 4,得到[2,5]范围内的随机数4。
#include <stdlib.h>
#include <time.h>
#include <math.h>

// 交换两个数的值
void swap(int *a, int *b) {
    int temp = *a;
    *a = *b;
    *b = temp;
}

// 选取随机主元,对区间[left,right]进行划分
int randPartition(int left, int right, int A[]) {
    // 生成随机数种子
    srand((unsigned) time(NULL));
    // 生成[left,right]范围内的随机数p
    int p = (int) round((1.0) * rand() / RAND_MAX * (right - left) + left);
    // 交换A[left]和A[p]
    swap(&A[left], &A[p]);
    
    // 以下为原先partition函数的划分过程,不需要改变任何东西
    int temp = A[left];			// 将A[left]存放至临时变量temp
    while (left < right) {		// left == right,即枢轴位置
        while (left < right && temp < A[right]) right--;	// 反复左移right
        A[left] = A[right];		// 将A[right]挪到A[left]
        while (left < right && A[left] <= temp) left++;		// 反复右移left
        A[right] = A[left];		// 将A[left]挪到A[right]
    }
    A[left] = temp;				// 把temp放到left与right相遇的地方
    return left;				// 返回相遇的下标
}

// 快速排序,left与right初值为序列首尾下标(如1与n)
void quick_sort(int left, int right, int A[]) {
    if (left < right) {                             // left == right时,区间只有一个元素
        int pivot = randPartition(left, right, A);	// 将[left,right]按A[left]一分为二
        quick_sort(left, pivot - 1, A);         	// 对左子区间递归进行快速排序
        quick_sort(pivot + 1, right, A);        	// 对右子区间递归进行快速排序
    }
}
  • 5
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值