hezhao.jpg

前言:排序按照所占用的计算机内部存储设备,可以分为:内部排序外部排序

  • 内部排序:占用的是内存,待排序序列全部放在内存加以排序处理
  • 外部排序:占用的是外存,数据量比较大,内存空间不足以一次性全部容纳数据的情况

本文章 通过912. 排序数组 题目,以此来总结内部排序的各种排序算法。

image.png

一、插入类排序

将无序的子序列插入到有序序列中

✅直接插入

各类排序算法汇总 - 图3
将元素序列走一遍,走到某个元素时,将其插入到已走过的已排序序列中,这样可以保证走完所有元素,然后所有的元素都是排序好的。
数据结构选用的时顺序表

  1. /**
  2. * @param {number[]} nums
  3. * @return {number[]}
  4. */
  5. var sortArray = function(nums) {
  6. for (let i = 0; i<nums.length; i++) {
  7. let flag = i
  8. for(let j = flag-1;j>=0;j--) {
  9. if (nums[flag] < nums[j]) {
  10. let temp = nums[j]
  11. nums[j] = nums[flag]
  12. nums[flag] = temp
  13. flag--
  14. }
  15. }
  16. }
  17. return nums
  18. };

image.png

优化:折半插入

在直接插入的过程中,找到一个元素,然后再需要从后往前依次查找“该在”的位置,对其查找进行了折半优化

  1. /* 折半插入排序 */
  2. void BinsertSort(SqList &S) {
  3. for (int i = 2; i <= S.length;i++) {
  4. S.data[0] = S.data[i];
  5. int low = 1;
  6. int high = i - 1;
  7. while (low <= high) {
  8. int m = (low + high) / 2;
  9. if (S.data[0]<S.data[m]) high = m - 1;
  10. else low = m + 1;
  11. }
  12. int j;
  13. for (j = i - 1; j >= high + 1;--j)
  14. S.data[j + 1] = S.data[j];
  15. S.data[high + 1] = S.data[0];
  16. }
  17. }

优化:希尔排序

各类排序算法汇总 - 图5

  1. /* 希尔排序 */
  2. void ShellInsert (SqList &L, int dk) {
  3. for (int i = dk + 1; i <= L.length;++i) {
  4. if (L.data[i]<L.data[i-dk]) {
  5. L.data[0] = L.data[i];
  6. int j;
  7. for (j = i - dk; j > 0 && L.data[0] < L.data[j]; j -= dk)
  8. L.data[j + dk] = L.data[j];
  9. L.data[j + dk] = L.data[0];
  10. }
  11. }
  12. }
  13. void ShellSort (SqList &L, int dt[],int t) {
  14. for (int k = 0; k < t;++k) {
  15. ShellInsert(L, dt[k]);
  16. }
  17. }

二、交换类排序

✅冒泡排序

各类排序算法汇总 - 图6

  1. /**
  2. * @param {number[]} nums
  3. * @return {number[]}
  4. */
  5. var sortArray = function(nums) {
  6. for (let i = nums.length - 1; i>0; i--) {
  7. for (let j = 0; j<i; j++) {
  8. if (nums[j] > nums[j+1]) {
  9. let temp = nums[j]
  10. nums[j] = nums[j+1]
  11. nums[j+1] = temp
  12. }
  13. }
  14. }
  15. return nums
  16. };

image.png

快速排序

各类排序算法汇总 - 图8

  1. /* 快速排序 */
  2. int Partition (SqList &L, int low, int high) {
  3. L.data[0] = L.data[low];
  4. int pivotkey = L.data[low];
  5. while (low < high) {
  6. while (low<high && L.data[high]>=pivotkey)
  7. --high;
  8. L.data[low] = L.data[high];
  9. while (low<high && L.data[low]<=pivotkey)
  10. ++low;
  11. L.data[high] = L.data[low];
  12. }
  13. L.data[low] = L.data[0];
  14. return low;
  15. }
  16. void Qsort(SqList &L, int low, int high) {
  17. if (low<high){
  18. int pivoloc = Partition(L, low, high);
  19. Qsort(L, low, pivoloc - 1);
  20. Qsort(L, pivoloc + 1, high);
  21. }
  22. }
  23. void QuickSort(SqList &L) {
  24. Qsort(L, 1, L.length);
  25. }

三、选择类排序

参考:ata.biancheng.net/view/72.html

✅简单选择排序

各类排序算法汇总 - 图9

  1. /**
  2. * @param {number[]} nums
  3. * @return {number[]}
  4. */
  5. var sortArray = function(nums) {
  6. for (let i = 0; i<nums.length-1; i++) {
  7. for (let j = i+1; j<nums.length; j++) {
  8. if (nums[i] > nums[j]) {
  9. let temp = nums[i]
  10. nums[i] = nums[j]
  11. nums[j] = temp
  12. }
  13. }
  14. }
  15. return nums
  16. };

image.png

树形选择排序

堆排序

各类排序算法汇总 - 图11

四、归并排序

2-路归并排序

给定一个序列,从左往右两两子序列进行归并

子序列归并的算法:加入有两个靠着的a、b序列,由上面可知,a、b各自都是有序序列,现在就是将这两个合并为一个有序序列k,将a和b序列的各个元素进行比较,小的依次放入k序列,当a、b两个中有一个序列为空了,就将那个不为空的序列直接加入到k序列即可,最后k序列就是目的序列。

五、分配类排序

基数排序

TODO: 扑克牌的花色排序

六、外部排序

基本方法

多路平衡归并

基本思想是内部排序中的2-路归并排序

置换-选择排序

最佳归并树

参考

文章的动画演示