IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 【快速排序】:绝对是史上最全的快速排序分析,分析了所有的边界情况并给出算法可视化动图帮助你理解算法 -> 正文阅读

[数据结构与算法]【快速排序】:绝对是史上最全的快速排序分析,分析了所有的边界情况并给出算法可视化动图帮助你理解算法

目录

一:快速排序概念定义

二:题目描述

三:算法详解

do i++; while(q[i] < x);? ?//用while(q[++i]= x)也是等价的语句>

do j--; while(q[j] > x);? 会使得 q[j+1..r] >= x, q[j] <= x

if(i < j) swap(q[i], q[j]);? 会使得 q[l..i] <= x, q[j..r] >= x

四.常见边界问题汇总以及分析

五.给出以i划分的模板

六.顺便给出从大到小快排的模板(只需要改动两个大小符号即可)

七:排序的稳定性以及时空复杂度


一:快速排序概念定义

????????排序问题可以说是最经典的算法问题了,一般一开始学的基础算法就是排序,排序既是算法的基础,也是考研数据结构的重要知识点。但是?C/C++ 都有现成的库函数(C里是 qsort、C++是sort),所以导致很多人就没有学排序算法本身,其实排序算法的思路非常有意思,从最简单的学起,对以后学习复杂算法是非常有帮助的,比如快速排序和快速选择之间的关系,归并排序和逆序对之间的关系。
??从今天开始,我们来试着掌握每个排序算法的思路,今天是快速排序。
??快速排序,采用的是分治的思想,选择一个基准点,把所有小于基准点的数放到它左边,把所有大于基准点的数放到它右边,如图所示:

图示含义
■ 蓝色的柱形表示还未排序的数
■ 黄色的柱形表示随机选定的基准数
■ 橙色的柱形表示已经排好序的数
■ 红色的柱形表示正在遍历比较的数
■ 绿色的柱形表示比基准数小的数
■ 紫色的柱形表示比基准数大的数

原始数据:

dfb5c79b1dbb4de6907b860224b00f87.png

排序可视化过程:

ae2de55c0521425bbdd66119b2625341.gif

?容易发现:第一次遍历选择的基准数是29,第一遍排序结束的结果是14,10,14,29,37,也就是29左边的数都小于等于29,29右边的数都大于等于29,符合我们对快速排序算法的定义,在此基础上递归排序即可


二:题目描述

? ? ? ? 给你一个长度为n(n<=100000)的整数数组a,请你将该数组升序排列


三:算法详解

先给出我个人的万能模板(个人习惯以j分界,文章后面详细说明怎么用i分界):?

#include <bits/stdc++.h>
using namespace std;

const int N=100010;

int q[N];

void quicksort (int q[],int l, int r)
{
    if (l>=r) return ;
    int i=l-1,j=r+1,x=q[(l+r)/2];
    while(i<j)
    {
        do i++;while(q[i]<x);
        do j--; while(q[j]>x);
        if(i<j) swap(q[i],q[j]);
    }
    
    quicksort(q,l,j);
    quicksort(q,j+1,r);
}


int main()
{
    int n;
    scanf("%d",&n);
    
    for(int i=0;i<n;i++)
    {
        scanf("%d",&q[i]);
    }
    
    quicksort(q,0,n-1);
    
    for(int  i=0;i<n;i++)
    {
    printf("%d ",q[i]);
    }
}

算法证明:?借鉴算法导论循环不变式的方法

快排核心模板
1.递归出局
2.处理某个子问题
3.递归处理子问题

//循环体

void quicksort(int q[], int l, int r)
{
? ? //第一步:规定递归的终止情况
? ? if(l >= r) return;
? ? //第二步:分成子问题,注意i和l的初始位置,待会详细说明为什么这么设置i和j和x
? ? int i = l - 1, j = r + 1, x = q[l + r >> 1];
? ? while(i < j)
? ? {
? ? ? ? do i++; while(q[i] < x);//注意不能有等于,待会详细说明为什么不能有等于
? ? ? ? do j--; while(q[j] > x);
? ? ? ? if(i < j) swap(q[i], q[j]);
? ? }
? ? //第三步:以j为分界,划分区间,递归处理子问题
? ? quick_sort(q, l, j), quick_sort(q, j + 1, r);
}

证明目标:循环不变式,即任意一次while循环(除了最后一次while循环)结束后,q[l..i] <= x,q[j..r] >= x

证明过程:

1.初始化:i = l - 1, j = r + 1,显然此时q[l..i],q[j..r]为空,证明目标显然成立

2.保持:假设下一轮循环开始之前的循环不变式成立,即?q[l..i] <= x? ,? q[j..r] >= x

执行上述的循环体

do i++; while(q[i] < x);? ?//用while(q[++i]<x)也是等价的语句
? 会使得 q[l..i-1] <= x, q[i] >= x

do j--; while(q[j] > x);
? 会使得 q[j+1..r] >= x, q[j] <= x

if(i < j) swap(q[i], q[j]);
? 会使得 q[l..i] <= x, q[j..r] >= x

所以,经过循环体之后,循环不变式依然成立?

注:由于i和j的设定问题,i和j必须要先自增!!否则在特殊情况(比如去q[i]和q[j]都等于x)下就会死循环

比如:

?while(q[i] < x)? i++;
? while(q[j] > x)? j--;
当q[i]和q[j]都为 x 时, i 和 j 都不会更新,导致 while 陷入死循环

3.终止:最后一轮循环结束时?j <=i且只有j+1=r或者j=r两种情况


四.常见边界问题汇总以及分析

1.以j划分时,随机值x为什么不能选q[r],以i划分时,随机值x为什么不能选q[l]

证明:假设x取q[r],关键问题在于递归子问题 quick_sort(q, l, j), quick_sort(q, j + 1, r);举例来说,若x选为q[r],且数组中q[l..r-1] < x,那么一轮循环结束时,i = r, j = r,quick_sort(q, l, j)直接退出函数调用,没有排序,以i划分也是同理

2.?do i++; while(q[i] < x)和do j--; while(q[j] > x)为什么不能用q[i] <= x 和 q[j] >= x

证明:假设q[l..r]全相等,则执行完do i++; while(q[i] <= x);之后,i会自增到r+1,然后继续执行q[i] <= x 判断条件,造成数组下标越界

3.?if(i < j) swap(q[i], q[j])能否使用 i <= j

证明:可以。因为 i = j 时,交换一下q[i],q[j] 无影响,因为马上就会跳出循环了

4.最后一句能否改用左区间quick_sort(q, l, j-1),右区间?quick_sort(q, j, r)作为划分(用i做划分时也有同样的问题)?

证明:不能。

情况一:i=j+1时,显然根据上述循环体的结论同意得到q[l,..j]<=x,q[i,..r](即q[j+1,r])>=x,因此我的原代码的划分一定正确。但是倘若使用上述错误的划分,q[l,..j-1]<=x,没有问题,但是q[j]<=x,却被递归在右半个区间,显然错误

情况二:i=j时,特殊情况i=j=1时,此时很容易得到q[i]=q[j]=x,倘若用上述的错误代码,左边区间调用函数会直接退出,但是右边区间等价于原来的情况原地踏步,所以该划分方式会导致死循环

五.给出以i划分的模板

void quick_sort(int q[], int l, int r)
{
    if(l >= r) return;

    int i = l - 1, j = r + 1, x = q[l + r + 1 >> 1];//注意是向上取整,因为向下取整可能使得x取到q[l],这是和j划分不一样的地方
    while(i < j)
    {
        do i++; while(q[i] < x);
        do j--; while(q[j] > x);
        if(i < j) swap(q[i], q[j]);
    }
    quick_sort(q, l, i - 1), quick_sort(q, i, r);//不用q[l..i],q[i+1..r]划分的道理和分析4中j的情况一样
}

六.顺便给出从大到小快排的模板(只需要改动两个大小符号即可)

void quicksort(int q[], int l, int r)
{
    if(l >= r) return;

    int i = l - 1, j = r + 1, x = q[l + r >> 1];
    while(i < j)
    {
        do i++; while(q[i] > x); // 这里和下面
        do j--; while(q[j] < x); // 这行的判断条件改一下
        if(i < j) swap(q[i], q[j]);
    }
    quicksort(q, l, j), quicksort(q, j + 1, r);
}

如果觉得有问题欢迎私聊,如果绝对写的不错,欢迎支持,创作不易。

七:排序的稳定性以及时空复杂度

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-08-06 11:07:37  更:2022-08-06 11:11:44 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年5日历 -2024/5/7 10:39:14-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码