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

排序算法 -- 统一视角

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

问题求解流程

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

本系列文章合集:


选择排序

子问题定义

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)

组合子问题的解


相关推荐
皮皮哎哟5 小时前
数据结构:嵌入式常用排序与查找算法精讲
数据结构·算法·排序算法·二分查找·快速排序
堕2745 小时前
java数据结构当中的《排序》(一 )
java·数据结构·排序算法
long3167 小时前
Aho-Corasick 模式搜索算法
java·数据结构·spring boot·后端·算法·排序算法
YuTaoShao9 小时前
【LeetCode 每日一题】3634. 使数组平衡的最少移除数目——(解法一)排序+滑动窗口
算法·leetcode·排序算法
庄周迷蝴蝶1 天前
四、CUDA排序算法实现
算法·排序算法
遨游xyz1 天前
排序-快速排序
开发语言·python·排序算法
历程里程碑1 天前
普通数组----最大子数组和
大数据·算法·elasticsearch·搜索引擎·排序算法·哈希算法·散列表
2302_813806221 天前
【嵌入式修炼:数据结构篇】——单向链表的排序
数据结构·链表·排序算法
酉鬼女又兒1 天前
27. 移除元素
数据结构·算法·排序算法
404未精通的狗1 天前
(数据结构)排序
数据结构·排序算法