剑指Offer-快速排序
剑指Offer上的快速排序的Partition函数与我在数据结构书上学到的不一样,因此就想要探索下这两种不同的处理方式。
基本思想
快速排序的基本思想是基于分治法。在待排序表L[1...n]中任取一个元素pivot作为基准,通过一趟排序将待排序表划分为独立的两部分L[1...k-1]和L[k+1...n],使得L[1...k-1]中所有元素小于pivot,L[k+1...n]中所有元素大于或等于pivot,则pivot放在了其最终位置L(k)上。而后递归地对两个子表重复上述过程,直至每部分内只有一个元素或空为止,即所有元素放在了其最终位置上。
剑指Offer上的Partition实现
// Partition为分割函数, length表示data数组的长度
int Partition(int data[], int length, int low, int high) {
if (data == nullptr || length <= 0 || low < 0 || high >= length) {
return 1;
}
// 在[low, high]区间中随机取一个基准值,并将其换到区间最末尾
int index = RandomInRange(low, high);
Swap(&data[index], &data[high]);
// small在low的前面一位
int small = low - 1;
for (index = low; index < high; ++index) {
if (data[index] < data[high]) {
// ++small后表示大于基准值的第一个数的下标
// 如果不存在大于基准值的数,则表示当前比较的数字下标
++small;
// 交换后的small表示小于基准值的最后一个数字的下标
if (small != index) {
Swap(&data[index], &data[small]);
}
}
}
// ++small后表示大于基准值的第一个数的下标
++small;
Swap(&data[small], &data[high]);
return small;
}
快速排序其它函数实现如下:
// 生成一个随机数,范围在[low,high),是一个前闭后开的区间
int RandomInRange(int low, int high) {
return rand() % (high - low) + low;
}
// 交换两个整数
void Swap(int* left, int* right) {
int temp = *left;
*left = *right;
*right = temp;
}
// 快速排序函数
void QuickSort(int data[], int length, int low, int high) {
if (low == high) {
return;
}
int index = Partition(data, length, low, high);
if (index > low) {
QuickSort(data, length, low, index - 1);
}
if (index < high) {
QuickSort(data, length, index + 1, high);
}
}
// 数据打印
void PrintData(int data[], int length) {
for (int i = 0; i < length; ++i) {
std::cout << data[i] << " ";
}
std::cout << std::endl;
}
int main() {
// 以系统时间作为随机函数种子
srand((unsigned)time(nullptr));
int const length = 10;
int data[length];
// 生成随机数
for (int i = 0; i < length; ++i) {
data[i] = RandomInRange(1, length*length);
}
// 打印生成的随机数
PrintData(data, length);
// 调用快排函数进行排序
QuickSort(data, length, 0, length - 1);
// 打印快速排序后的数
PrintData(data, length);
return 0;
}
初始顺序为{49, 38, 65, 97, 76, 13, 27},基准值为49的数据的第一趟快速排序的实现如下所示:
步骤 | small | index | |||||||
---|---|---|---|---|---|---|---|---|---|
0 | 49 | 38 | 65 | 97 | 76 | 13 | 27 | ||
1 | 27 | 38 | 65 | 97 | 76 | 13 | 49 | ||
2 | 0 | 0 | 27 | 38 | 65 | 97 | 76 | 13 | 49 |
3 | 1 | 1 | 27 | 38 | 65 | 97 | 76 | 13 | 49 |
4 | 1 | 2 | 27 | 38 | 65 | 97 | 76 | 13 | 49 |
5 | 1 | 3 | 27 | 38 | 65 | 97 | 76 | 13 | 49 |
6 | 1 | 4 | 27 | 38 | 65 | 97 | 76 | 13 | 49 |
7 | 2 | 5 | 27 | 38 | 13 | 97 | 76 | 65 | 49 |
8 | 3 | 6 | 27 | 38 | 13 | 49 | 76 | 65 | 97 |
第一趟排序完之后基准值49之前的{27, 38, 13}都比49小,基准值49之后的{76, 65, 97}都比49大。
一般教材上的Partition实现
// Partition为分割函数
int Partition(int data[], int low, int high) {
// 将当前表中第一个元素设为基准值,对表进行划分
int pivot = data[low];
// 循环跳出条件
while (low < high) {
while (low < high && data[high] >= pivot) {
--high;
}
// 将比基准值小的元素移动到左端
data[low] = data[high];
while (low < high && data[low] <= pivot) {
++low;
}
// 将比基准值大的元素移动到右端
data[high] = data[low];
}
// 基准元素存放到最终位置
data[low] = pivot;
// 返回存放基准值的最终位置
return low;
}
快速排序其它函数实现如下:
void QuickSort(int data[], int low, int high) {
// 递归跳出条件
if (low < high) {
// 将data[low...high]一分为二,pivotpos是基准值
int pivotpos = Partition(data, low, high);
// 对左子表进行递归排序
QuickSort(data, low, pivotpos - 1);
// 对右子表进行递归排序
QuickSort(data, pivotpos + 1, high);
}
}
初始顺序为{49, 38, 65, 97, 76, 13, 27},基准值为49的数据的第一趟快速排序的实现如下所示:
步骤 | low | high | |||||||
---|---|---|---|---|---|---|---|---|---|
0 | 0 | 6 | 49 | 38 | 65 | 97 | 76 | 13 | 27 |
1 | 2 | 6 | 27 | 38 | 65 | 97 | 76 | 13 | 27 |
2 | 2 | 6 | 27 | 38 | 65 | 97 | 76 | 13 | 65 |
3 | 2 | 5 | 27 | 38 | 13 | 97 | 76 | 13 | 65 |
4 | 3 | 5 | 27 | 38 | 13 | 97 | 76 | 97 | 65 |
5 | 3 | 3 | 27 | 38 | 13 | 49 | 76 | 97 | 65 |
第一趟排序完之后基准值49之前的{27, 38, 13}都比49小,基准值49之后的{76, 97, 65}都比49大。
总结
可以看出,上述这两种思想导致的运行过程是不同的,前者是用small指代小于标准元素的最右位置,在遍历过程中进行交换操作来保证small左边都是小于标准元素的元素,而后者是用low,high分别表示小于和大于基准值的集合边界位置,所以后者更浅显易懂。
个人主页:
剑指Offer-快速排序的更多相关文章
- 剑指 Offer 40. 最小的k个数 + 优先队列 + 堆 + 快速排序
剑指 Offer 40. 最小的k个数 Offer_40 题目描述 解法一:排序后取前k个数 /** * 题目描述:输入整数数组 arr ,找出其中最小的 k 个数.例如,输入4.5.1.6.2.7. ...
- [读]剑指offer
研二的开始找工作了,首先祝愿他们都能够找到自己满意的工作.看着他们的身影,自问明年自己这个时候是否可以从容面对呢?心虚不已,赶紧从老严那儿讨来一本<剑指offer>.在此顺便将自己做题所想 ...
- 剑指OFFER——调整数组顺序使奇数位于偶数前面
输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于位于数组的后半部分,并保证奇数和奇数,偶数和偶数之间的相对位置不变. 剑指offer书里的版本, ...
- 剑指Offer——顺丰笔试题+知识点总结
剑指Offer--顺丰笔试题+知识点总结 情景回顾 时间:2016.10.16 19:00-20:40 地点:山东省网络环境智能计算技术重点实验室 事件:顺丰笔试 知识点总结 快排 霍尔排序(快排) ...
- 剑指Offer——知识点储备-常用算法
剑指Offer--知识点储备-常用算法 快速排序 注:若排序是有序的,采用快排,则退化为冒泡排序. 解决这个问题,采用两个选取基准的方法 (1)随机选取基数(在这个区间内随机取一个数) 出现的恶劣情况 ...
- 剑指Offer——好未来视频面知识点储备+面后总结
剑指Offer--好未来视频面知识点储备+面后总结 情景介绍 时间:2016.10.12 13:00- 地点:宿舍 事件:好未来视频面 知识点储备 数据结构 单链表反转 public class Li ...
- 剑指Offer——分治算法
剑指Offer--分治算法 基本概念 在计算机科学中,分治法是一种很重要的算法.字面上的解释是"分而治之",就是把一个复杂的问题分成两个或更多的相同或相似的子问题,再把子问题分成更 ...
- 剑指offer:调整数组顺序使奇数位于偶数前面
题目 输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有奇数位于数组的前半部分,所有偶数位于数组的后半部分. 分析 事实上,这个题比较简单,很多种方式都可以实现,但是其时间复杂度或空间复 ...
- 关于数据结构,剑指offer上面的
我很喜欢那些javascript解决的编程题,感觉非常的有意思.我在博客园上面看到了一个同学的博客,他一共发了34篇剑指offer的编程题,还给出了非常详细的解答. 接下来的工作,我做的就是搬运工,不 ...
- 《剑指offer》内容总结
(1)剑指Offer——Trie树(字典树) Trie树 Trie树,即字典树,又称单词查找树或键树,是一种树形结构,是一种哈希树的变种.典型应用是统计和排序大量的字符串(但不仅限于字符串),所以经常 ...
随机推荐
- mysql——查询语句——单表查询——(概念)
一.基本查询语句 select的基本语法格式如下: select 属性列表 from 表名和视图列表 [ where 条件表达式1 ] [ group by 属性名1 [ having 条件表达式2 ...
- 树莓派4B 串口通信
提前下载安装Glade图形编辑器 参考 树莓派4B安装netcore 环境部署.发布.执行操作 准备串口设备本文使用串口控制继电器设备 如图 1.发现串口 void GetSerialPort() { ...
- kafka服务器批量copy文件脚本
#!/bin/bashif [[ $# -lt 1 ]] ; then echo no params ; exit ; fiparams=$@for (( i=1 ; i <= 3 ; i = ...
- 从零开始,SpreadJS新人学习笔记【第4周】
数据绑定.脏数据和单引号前缀 本周,让我们一起来学习SpreadJS 的数据绑定.脏数据和单引号前缀,希望我的学习笔记能够帮助你们,从零开始学习 SpreadJS,并逐步精通. 在此前的学习笔记中,相 ...
- kafka整理笔记笔记
一.为什么需要消息系统 解耦: 允许你独立的扩展或修改两边的处理过程,只要确保它们遵守同样的接口约束. 冗余: 消息队列把数据进行持久化直到它们已经被完全处理,通过这一方式规避了数据丢失风险.许多消息 ...
- Spring_four
Spring_four 基于XML的AOP实现事务控制 坐标xml ; //2.6更新转入账户 accountDao.updateAccount(target); } } 注意:方法级别的事务会覆盖类 ...
- 什么是云数据库 HBase 版
云数据库 HBase 版(ApsaraDB for HBase)是基于 Hadoop 的一个分布式数据库,支持海量的PB级的大数据存储,适用于高吞吐的随机读写的场景.目前在阿里内部有数百个集群,100 ...
- C++多线程基础学习笔记(一)
下面分三个方面多线程技术的必须掌握一些基本知识. 1.进程 2.线程 3.并发 (1)进程 一个可执行程序运行起来了,即为创建了一个进程.如在电脑上打开了word,就创建了一个word进程,打开QQ, ...
- Django基础之模型(models)层(上)
目录 Django基础之模型(models)层 单表查询 必知必会13条 神奇的双下划线查询 多表查询 外键的字段的增删改查 表与表之间的关联查询 基于双下划线的跨表查询(连表查询) 补充知识 Dja ...
- Python之模块IO
目录 Python之模块IO io概叙 io类层次结构 io模块的类图 io模块的3种I/O 原始I/O,即RawIOBase及其子类 文本I/O,即TextIOBase及其子类 字节I/O(缓存I/ ...