数据结构与算法篇-排序算法-统一视角

排序算法 -- 统一视角

本系列文章尝试从"问题拆分+基准情况+组合解"这个统一视角出发,理解各种排序。

问题求解流程

  • 第一步:原问题怎么拆分?
  • 第二步:最小的子问题是什么?
  • 第三步:如何组合子问题的解得到原问题的解?

本系列文章合集:


选择排序

子问题定义

selectionSort(arr, start):表示当前要排序的子区间是arr[start...n-1]

初始子问题(原问题)

selectionSort(arr, 0):表示当前要排序的子区间是arr[0...n-1]

基准情况

start >= n-1表示当前要排序的子区间是arr[start...n-1]的长度为 0 或者 1。因为单个元素或者空集是有序的,所以该问题已解决,不用继续拆分了。

问题拆分

拆分前:selectionSort(arr, start)

目标:保证位置start对应的元素已在其最终位置上。

操作:从左向右遍历未排序区间,找到最小值,与未排序区间的首个元素交换

拆分后:selectionSort(arr, start+1)

组合子问题的解


冒泡排序

子问题定义

bubbleSort(arr, start):表示当前要排序的子区间是arr[start...n-1]

初始子问题(原问题)

bubbleSort(arr, 0):表示当前要排序的子区间是arr[0...n-1]

基准情况

start >= n-1表示当前要排序的子区间是arr[start...n-1]的长度为 0 或者 1。因为单个元素或者空集是有序的,所以该问题已解决,不用继续拆分了。

问题拆分

拆分前:bubbleSort(arr, start)

目标:保证位置start对应的元素已在其最终位置上。

操作:从右向左遍历未排序区间arr[start...n-1],不断比较相邻元素,如果逆序就交换,最终使得最小值冒泡到该区间的左侧。

拆分后:bubbleSort(arr, start+1)

组合子问题的解


插入排序

子问题定义

insertSort(arr, start)

  • 表示当前要排序的子区间是arr[start...n-1]
  • 区间arr[0...start-1]已有序

初始子问题(原问题):
insertSort(arr, 1)

  • 表示当前要排序的子区间是arr[1...n-1]
  • 区间arr[0...0]已有序,因为此时该区间只有 1 个元素。

基准情况

start >= n表示当前要排序的子区间是arr[n...n-1]的长度为 0 ,且区间arr[0...start-1]已有序。因为空集是有序的,所以该问题已解决,不用继续拆分了。

问题拆分

拆分前:insertSort(arr, start)

目标:保证位置start对应的元素已在其最终位置上。

操作:取未排序区间arr[start...n-1]的首个元素arr[start],从已有序区间arr[0...start-1]中找到其正确的插入位置,将其插入。

拆分后:insertSort(arr, start+1)

组合子问题的解


归并排序

子问题定义

mergeSort(arr, left, right):表示当前要排序的子区间是arr[left...right]

初始子问题(原问题)

mergeSort(arr, 0, n-1):表示当前要排序的子区间是arr[0...n-1]

基准情况

left >= right:表示当前要排序的子区间是arr[left...right],它的长度为 1或者 0,因为单个元素或者空集是有序的,不用继续拆分了。

问题拆分

拆分前:mergeSort(arr, left, right)

操作:取区间的中点mid = left + (right-left)/2

拆分后:mergeSort(arr, left, mid)mergeSort(arr, mid+1, right)

组合子问题的解

合并,即两个有序的子区间arr[left...mid]arr[mid+1...right]合并成一个整体有序的区间arr[left...right]


堆排序

子问题定义

heapSort(arr, heapSize)

  • 当前要排序的子区间是arr[0...heapSize-1]

区间划分状态:

  • 区间arr[0...heapSize-1] 已经是一个合法的堆
  • 区间arr[heapSize..n-1]已有序,且每个元素都在最终位置上

初始子问题(原问题)

heapSort(arr, n)

  • 当前要排序的子区间是arr[0...n-1]

区间划分状态:

  • 区间arr[0...n-1] 已经是一个合法的堆
  • 区间arr[n..n-1]已有序,且每个元素都在最终位置上

基准情况

heapSize<=1

heapSort(arr, 1)

  • 当前要排序的子区间是arr[0...0]

区间划分状态:

  • 区间arr[0...0] 已经是一个合法的堆
  • 区间arr[1..n-1]已有序,且每个元素都在最终位置上

问题拆分

拆分前:heapSort(arr, heapSize)

目标:将当前堆区间的最大值放在其最终位置heapSize-1

操作:

  • 将当前堆区间arr[0, heapSize-1]的最大值跟arr[heapSize-1]交换
  • heapSize--;
  • 对缩小后的堆区间arr[0,heapSize-1]执行下沉操作,以恢复堆性质

拆分后:heapSort(arr, heapSize-1)

组合子问题的解


相关推荐
bbbb3651 小时前
排序算法的演进史:从冒泡到快速再到TimSort的技术7
数据结构·算法·排序算法
不染尘.1 小时前
排序算法详解1
开发语言·数据结构·c++·算法·排序算法
爱喝一杯白开水1 小时前
Java List 常用方法全攻略
java·list·排序算法
Elsa️7461 小时前
洛谷p5718 复习下快速排序和堆排序
数据结构·算法·排序算法
myloveasuka14 小时前
[Java]查找算法&排序算法
java·算法·排序算法
We་ct14 小时前
LeetCode 148. 排序链表:归并排序详解
前端·数据结构·算法·leetcode·链表·typescript·排序算法
abant221 小时前
leetcode912 排序算法总结
算法·leetcode·排序算法
重生之后端学习1 天前
31. 下一个排列
数据结构·算法·leetcode·职场和发展·排序算法·深度优先
像污秽一样2 天前
算法设计与分析-习题9.2
数据结构·算法·排序算法·dfs
小王不爱笑1323 天前
排序算法 Java
数据结构·算法·排序算法