快速排序
2021-05-30 22:05
标签:越界 特殊情况 class 相等 影响 快速排序 第一步 ons 快排 算法证明使用算法导论里的循环不变式方法 快排属于分治算法,分治算法都有三步: while循环结束后, 注: 循环不变式: 循环开始之前 则 假设某轮循环开始前循环不变式成立,即 执行循环体 所以,i和j更新之后,下一次循环开始之前,循环不变式依然成立 注意:由于使用do-while循环,所以 例如: 终止 循环结束时, 正常情况下,按照循环不变式,我们应该会觉得结果已经显然了 因为 所以按照 可是,最后一轮循环有点特殊,因为最后一轮循环的if语句一定不会执行 因为最后一轮循环一定满足 i >= j,不然不会跳出while循环的,所以if语句一定不执行 正确分析: 由于最后一轮的if语句一定不执行 所以,只能保证 由 又因为 总结:只有最后一轮循环结束时,循环不变式不成立,其余的循环都是成立的 但最终要求的问题还是解决了 注意:循环结束时要记得检查是否存在数组越界/无限递归的情况 所以还需要证明 快排属于分治算法,最怕的就是 假设 关键句子 由于 但 举例来说,若 那么这一轮循环结束时 假设 则执行完 然后继续执行 并且如果之后的 就会造成一直循环下去(亲身实验),造成内存超限 可以使用 因为 最后一句能否改用 不能 根据之前的证明,最后一轮循环可以得到这些结论 所以, 但 另外一点,注意 当 如果手动改为 但是上面所说的 证明: 假设 说明 说明 说明 得出 反证法得出 假设 反证法得出 所以 顺带一提用 快速排序 标签:越界 特殊情况 class 相等 影响 快速排序 第一步 ons 快排 原文地址:https://www.cnblogs.com/mrmrwjk/p/14747495.html快速排序模板
快速排序算法的证明与边界分析
算法证明
快排模板(以j为分界)
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];
while(i x);
if(i
待证问题
q[l..j] ,
q[j+1..r] >= x
q[l..j] 意为
q[l],q[l+1]...q[j-1],q[j]
的所有元素都证明
q[l..i] = x
i = l - 1, j = r + 1
q[l..i],q[j..r]
为空,循环不变式显然成立
q[l..i] = x
do i++; while(q[i] = x
do j--; while(q[j] > x);
会使得 q[j+1..r] >= x, q[j] = x
复制代码
i
和j
一定会!!!自增!!!,使得循环会继续下去,但是如果采用while循环(i
和j
的初始化做出对应的变更),i
和j
在特殊情况下不自增的话,循环就会卡死 while(q[i] x) j--;
当q[i]和q[j]都为 x 时, i 和 j 都不会更新,导致 while 陷入死循环
复制代码
i >= j
i >= j,q[l..i] = x
j
来划分的话,q[l..j] = x
是显然的i >= j
和q[l..i-1] = x
和q[j+1..r] >= x, q[j]
q[l..i-1] = j(i-1 >= j-1)
和 q[j] 可以得到
q[l..j]
q[j+1..r] >= x
所以,q[l..j] = x
,问题得证j
最终的取值范围是[l..r-1]
(即不存在n
划分成0
和n
的无限递归情况),分析过程在分析5
边界情况分析
分析
n分成0和n,或 n分成n和0
,这会造成无限划分
j
为划分时,x
不能选q[r]
(若以i
为划分,则x
不能选q[l]
)x = q[r]
quick_sort(q, l, j), quick_sort(q, j + 1, r);
j
的最小值是l
,所以q[j+1..r]
不会造成无限划分q[l..j]
(即quick_sort(q, l, j))却可能造成无限划分,因为j可能为rx
选为q[r]
,数组中q[l..r-1] ,
i = r, j = r
,显然会造成无限划分
do i++; while(q[i] 和
do j--; while(q[j] > x)
不能用q[i] 和
q[j] >= x
q[l..r]
全相等do i++; while(q[i] 之后,
i
会自增到r+1
q[i] 判断条件,造成数组下标越界(但这貌似不会报错)
q[i] r)
条件也不幸成立,(Memory Limit Exceeded)
if(i 能否使用
i
if(i
i = j
时,交换一下q[i],q[j]
无影响,因为马上就会跳出循环了quick_sort(q, l, j-1), quick_sort(q, j, r)
作为划分(用i
做划分时也是同样的道理,)j 和
q[l..i-1] = x
和 q[j+1..r] >= x, q[j]
q[l..j-1] 是显然成立的,
quick_sort(q, j, r)
中的q[j]
却是 q[j] ,这不符合快排的要求
quick_sort(q, l, j-1), quick_sort(q, j, r)
可能会造成无线划分x
选为q[l]
时会造成无限划分,报错为(MLE),x = q[r]
,可以避免无限划分q[j] 的问题依然不能解决,这会造成
WA (Wrong Answer)
j
的取值范围为[l..r-1]
j
最终的值为 r
,说明只有一轮循环(两轮的话 j
至少会自减两次)q[r] (因为要跳出
do-while
循环)i >= r
(while循环的结束条件), i
为 r
或 r + 1
(必不可能成立)i
自增到了 r
, 说明 q[r] >= x 和 q[l..r-1] ,
q[r] = x 和 q[l..r-1] 的结论,但这与
x = q[l + r >> 1]
矛盾j
j
可能小于 l
说明 q[l..r] > x
,矛盾j >= l
j
的取值范围为[l..r-1]
,不会造成无限划分和数组越界
while
循环的结束条件能不能改成 i
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]
while(i x);
if(i
#include