算法图解——读书笔记04

快速排序

学习快速排序—— 一种常用的优雅的排序算法,快速排序使用分而治之的策略。

分而治之策略是什么?

分而治之(divideand conquer, D&C)——一种著名的递归式问题解决方法。D&C提供了解决问题的思路,是另一个可供你使用的工具。面对新问题时,你不再束手无策,而是自问:“使用分而治之能解决吗?”

D&C算法是递归的。使用D&C解决问题的过程包括两个步骤。
(1) 找出基线条件,这种条件必须尽可能简单。
(2) 不断将问题分解(或者说缩小规模),直到符合基线条件。

D&C并非可用于解决问题的算法,而是一种解决问题的思路。我们再来看一个例子。
给定一个数字数组。
在这里插入图片描述
你需要将这些数字相加,并返回结果。使用循环很容易完成这种任务。
在这里插入图片描述
但如何使用递归函数来完成这种任务呢?

第一步:找出基线条件。最简单的数组什么样呢?请想想这个问题,再接着往下读。如果数组不包含任何元素或只包含一个元素,计算总和将非常容易。
在这里插入图片描述
因此这就是基线条件。

第二步:每次递归调用都必须离空数组更近一步。如何缩小问题的规模呢?下面是一种办法。
在这里插入图片描述
这与下面的版本等效。
在这里插入图片描述
这两个版本的结果都为12,但在第二个版本中,给函数sum传递的数组更短。换言之,这缩小了问题的规模!

函数sum的工作原理类似于下面这样。
在这里插入图片描述
这个函数的运行过程如下。
在这里插入图片描述
别忘了,递归记录了状态。
在这里插入图片描述
快速排序的工作原理

包含三个元素的数组
在这里插入图片描述

  • 首先,从数组中选择一个元素,这个元素被称为基准值(pivot)。
  • 接下来,找出比基准值小的元素以及比基准值大的元素。
    在这里插入图片描述
    这被称为分区(partitioning)。现在你有:
    ❑ 一个由所有小于基准值的数字组成的子数组;
    ❑ 基准值;
    ❑ 一个由所有大于基准值的数组组成的子数组。

如何对子数组进行排序呢?对于包含两个元素的数组(左边的子数组)以及空数组(右边的子数组),快速排序知道如何将它们排序,因此只要对这两个子数组进行快速排序,再合并结果,就能得到一个有序数组!

不管将哪个元素用作基准值,这都管用。假设你将15用作基准值。
在这里插入图片描述
步骤如下。
(1) 选择基准值。
(2) 将数组分成两个子数组:小于基准值的元素和大于基准值的元素。
(3) 对这两个子数组进行快速排序。

包含四个元素的数组呢?
在这里插入图片描述
假设你也将33用作基准值。
在这里插入图片描述
左边的子数组包含三个元素,而你知道如何对包含三个元素的数组进行排序:对其递归地调用快速排序。
在这里插入图片描述
因此你能够对包含四个元素的数组进行排序。如果能够对包含四个元素的数组进行排序,就能对包含五个元素的数组进行排序。为什么呢?假设有下面这样一个包含五个元素的数组。
在这里插入图片描述
根据选择的基准值,对这个数组进行分区的各种可能方式如下。
在这里插入图片描述
注意,这些子数组包含的元素个数都在0~4内,而你已经知道如何使用快速排序对包含0~4个元素的数组进行排序!因此,不管如何选择基准值,你都可对划分得到的两个子数组递归地进行快速排序。

例如,假设你将3用作基准值,可对得到的子数组进行快速排序。
在这里插入图片描述
将子数组排序后,将它们合并,得到一个有序数组。即便你将5用作基准值,这也可行。
在这里插入图片描述
将任何元素用作基准值都可行,因此你能够对包含五个元素的数组进行排序。同理,你能够对包含六个元素的数组进行排序,以此类推。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值