categories: [Blog,Algorithm,mid]


912. 排序数组

难度中等
给你一个整数数组 nums,请你将该数组升序排列。

示例 1:
输入:nums = [5,2,3,1]
输出:[1,2,3,5]

示例 2:
输入:nums = [5,1,1,2,0,0]
输出:[0,0,1,1,2,5]

选择排序

每一轮选择最小元素交换到未排定部分的开头。ok

  1. // 选择排序:每一轮选择最小元素交换到未排定部分的开头
  2. public int[] sortArray(int[] nums) {
  3. int len = nums.length;
  4. // 循环不变量:[0, i) 有序,且该区间里所有元素就是最终排定的样子
  5. for (int i = 0; i < len - 1; i++) {
  6. // 选择区间 [i, len - 1] 里最小的元素的索引,交换到下标 i
  7. int minIndex = i;
  8. for (int j = i + 1; j < len; j++) {
  9. if (nums[j] < nums[minIndex]) {
  10. minIndex = j;//
  11. }
  12. }
  13. swap(nums, i, minIndex);
  14. }
  15. return nums;
  16. }
  17. private void swap(int[] nums, int index1, int index2) {
  18. int temp = nums[index1];
  19. nums[index1] = nums[index2];
  20. nums[index2] = temp;
  21. }


冒泡排序

比较两个相邻的元素,将值大的元素交换到右边【超时】。

  1. // 冒泡排序:比较两个相邻的元素,将值大的元素交换到右边。
  2. public int[] sortArray(int[] nums) {
  3. int len = nums.length;
  4. for (int i = 0; i < len -1 ; i++) { // [5,2,3,1] i索引到3即length-2时 j比较的是3,1 length-2,length-1.
  5. // 先默认数组是有序的,只要发生一次交换,就必须进行下一轮比较,
  6. // 如果在内层循环中,都没有执行一次交换操作,说明此时数组已经是升序数组
  7. boolean sorted = true;
  8. for (int j = 0; j < len-i-1 ; j++) {
  9. if (nums[j] > nums[j+1]) {
  10. swap(nums, j, j+1);
  11. sorted = false;
  12. }
  13. }
  14. if(sorted){
  15. break;
  16. }
  17. }
  18. return nums;
  19. }

堆排序

  1. public class Solution {
  2. public int[] sortArray(int[] nums) {
  3. int len = nums.length;
  4. // 将数组整理成堆
  5. heapify(nums);
  6. // 循环不变量:区间 [0, i] 堆有序
  7. for (int i = len - 1; i >= 1; ) {
  8. // 把堆顶元素(当前最大)交换到数组末尾
  9. swap(nums, 0, i);
  10. // 逐步减少堆有序的部分
  11. i--;
  12. // 下标 0 位置下沉操作,使得区间 [0, i] 堆有序
  13. siftDown(nums, 0, i);
  14. }
  15. return nums;
  16. }
  17. /**
  18. * 将数组整理成堆(堆有序)
  19. *
  20. * @param nums
  21. */
  22. private void heapify(int[] nums) {
  23. int len = nums.length;
  24. // 只需要从 i = (len - 1) / 2 这个位置开始逐层下移
  25. for (int i = (len - 1) / 2; i >= 0; i--) {
  26. siftDown(nums, i, len - 1);
  27. }
  28. }
  29. /**
  30. * @param nums
  31. * @param k 当前下沉元素的下标
  32. * @param end [0, end] 是 nums 的有效部分
  33. */
  34. private void siftDown(int[] nums, int k, int end) {
  35. while (2 * k + 1 <= end) {
  36. int j = 2 * k + 1;
  37. if (j + 1 <= end && nums[j + 1] > nums[j]) {
  38. j++;
  39. }
  40. if (nums[j] > nums[k]) {
  41. swap(nums, j, k);
  42. } else {
  43. break;
  44. }
  45. k = j;
  46. }
  47. }
  48. private void swap(int[] nums, int index1, int index2) {
  49. int temp = nums[index1];
  50. nums[index1] = nums[index2];
  51. nums[index2] = temp;
  52. }
  53. }

快排

  1. // 快速排序 1:基本快速排序
  2. /**
  3. * 列表大小等于或小于该大小,将优先于 quickSort 使用插入排序
  4. */
  5. private static final int INSERTION_SORT_THRESHOLD = 7;
  6. private static final Random RANDOM = new Random();
  7. public int[] sortArray(int[] nums) {
  8. int len = nums.length;
  9. quickSort(nums, 0, len - 1);
  10. return nums;
  11. }
  12. private void quickSort(int[] nums, int left, int right) {
  13. // 小区间使用插入排序
  14. if (right - left <= INSERTION_SORT_THRESHOLD) {
  15. insertionSort(nums, left, right);
  16. return;
  17. }
  18. int pIndex = partition(nums, left, right);
  19. quickSort(nums, left, pIndex - 1);
  20. quickSort(nums, pIndex + 1, right);
  21. }
  22. /**
  23. * 对数组 nums 的子区间 [left, right] 使用插入排序
  24. *
  25. * @param nums 给定数组
  26. * @param left 左边界,能取到
  27. * @param right 右边界,能取到
  28. */
  29. private void insertionSort(int[] nums, int left, int right) {
  30. for (int i = left + 1; i <= right; i++) {
  31. int temp = nums[i];
  32. int j = i;
  33. while (j > left && nums[j - 1] > temp) {
  34. nums[j] = nums[j - 1];
  35. j--;
  36. }
  37. nums[j] = temp;
  38. }
  39. }
  40. private int partition(int[] nums, int left, int right) {
  41. int randomIndex = RANDOM.nextInt(right - left + 1) + left;
  42. swap(nums, left, randomIndex);
  43. // 基准值
  44. int pivot = nums[left];
  45. int lt = left;
  46. // 循环不变量:
  47. // all in [left + 1, lt] < pivot
  48. // all in [lt + 1, i) >= pivot
  49. for (int i = left + 1; i <= right; i++) {
  50. if (nums[i] < pivot) {
  51. lt++;
  52. swap(nums, i, lt);
  53. }
  54. }
  55. swap(nums, left, lt);
  56. return lt;
  57. }
  58. private void swap(int[] nums, int index1, int index2) {
  59. int temp = nums[index1];
  60. nums[index1] = nums[index2];
  61. nums[index2] = temp;
  62. }

基本思路:快速排序每一次都排定一个元素(这个元素呆在了它最终应该呆的位置),然后递归地去排它左边的部分和右边的部分,依次进行下去,直到数组有序;
算法思想:分而治之(分治思想),与「归并排序」不同,「快速排序」在「分」这件事情上不想「归并排序」无脑地一分为二,而是采用了 partition 的方法(书上,和网上都有介绍,就不展开了),因此就没有「合」的过程。
实现细节(注意事项):(针对特殊测试用例:顺序数组或者逆序数组)一定要随机化选择切分元素(pivot),否则在输入数组是有序数组或者是逆序数组的时候,快速排序会变得非常慢(等同于冒泡排序或者「选择排序」);
复杂度分析:
时间复杂度:O(NlogN),这里 N 是数组的长度;
空间复杂度:O(logN),这里占用的空间主要来自递归函数的栈空间。


归并排序

  1. // 归并排序
  2. /**
  3. * 列表大小等于或小于该大小,将优先于 mergeSort 使用插入排序
  4. */
  5. private static final int INSERTION_SORT_THRESHOLD = 7;
  6. public int[] sortArray(int[] nums) {
  7. int len = nums.length;
  8. int[] temp = new int[len];
  9. mergeSort(nums, 0, len - 1, temp);
  10. return nums;
  11. }
  12. /**
  13. * 对数组 nums 的子区间 [left, right] 进行归并排序
  14. *
  15. * @param nums
  16. * @param left
  17. * @param right
  18. * @param temp 用于合并两个有序数组的辅助数组,全局使用一份,避免多次创建和销毁
  19. */
  20. private void mergeSort(int[] nums, int left, int right, int[] temp) {
  21. // 小区间使用插入排序
  22. if (right - left <= INSERTION_SORT_THRESHOLD) {
  23. insertionSort(nums, left, right);
  24. return;
  25. }
  26. int mid = left + (right - left) / 2;
  27. // Java 里有更优的写法,在 left 和 right 都是大整数时,即使溢出,结论依然正确
  28. // int mid = (left + right) >>> 1;
  29. mergeSort(nums, left, mid, temp);
  30. mergeSort(nums, mid + 1, right, temp);
  31. // 如果数组的这个子区间本身有序,无需合并
  32. if (nums[mid] <= nums[mid + 1]) {
  33. return;
  34. }
  35. mergeOfTwoSortedArray(nums, left, mid, right, temp);
  36. }
  37. /**
  38. * 对数组 arr 的子区间 [left, right] 使用插入排序
  39. *
  40. * @param arr 给定数组
  41. * @param left 左边界,能取到
  42. * @param right 右边界,能取到
  43. */
  44. private void insertionSort(int[] arr, int left, int right) {
  45. for (int i = left + 1; i <= right; i++) {
  46. int temp = arr[i];
  47. int j = i;
  48. while (j > left && arr[j - 1] > temp) {
  49. arr[j] = arr[j - 1];
  50. j--;
  51. }
  52. arr[j] = temp;
  53. }
  54. }
  55. /**
  56. * 合并两个有序数组:先把值复制到临时数组,再合并回去
  57. *
  58. * @param nums
  59. * @param left
  60. * @param mid [left, mid] 有序,[mid + 1, right] 有序
  61. * @param right
  62. * @param temp 全局使用的临时数组
  63. */
  64. private void mergeOfTwoSortedArray(int[] nums, int left, int mid, int right, int[] temp) {
  65. System.arraycopy(nums, left, temp, left, right + 1 - left);
  66. int i = left;
  67. int j = mid + 1;
  68. for (int k = left; k <= right; k++) {
  69. if (i == mid + 1) {
  70. nums[k] = temp[j];
  71. j++;
  72. } else if (j == right + 1) {
  73. nums[k] = temp[i];
  74. i++;
  75. } else if (temp[i] <= temp[j]) {
  76. // 注意写成 < 就丢失了稳定性(相同元素原来靠前的排序以后依然靠前)
  77. nums[k] = temp[i];
  78. i++;
  79. } else {
  80. // temp[i] > temp[j]
  81. nums[k] = temp[j];
  82. j++;
  83. }
  84. }
  85. }

基本思路:借助额外空间,合并两个有序数组,得到更长的有序数组。例如:「力扣」第 88 题:合并两个有序数组。
算法思想:分而治之(分治思想)。「分而治之」思想的形象理解是「曹冲称象」、MapReduce,在一定情况下可以并行化。
个人建议:「归并排序」是理解「递归思想」的非常好的学习材料,大家可以通过理解:递归完成以后,合并两个有序数组的这一步骤,想清楚程序的执行流程。即「递归函数执行完成以后,我们还可以做点事情」。因此,「归并排序」我个人觉得非常重要,一定要掌握。


插入排序

  1. // 插入排序:稳定排序,在接近有序的情况下,表现优异
  2. public int[] sortArray(int[] nums) {
  3. int len = nums.length;
  4. // 循环不变量:将 nums[i] 插入到区间 [0, i) 使之成为有序数组
  5. for (int i = 1; i < len; i++) {
  6. // 先暂存这个元素,然后之前元素逐个后移,留出空位
  7. int temp = nums[i];
  8. int j = i;
  9. // 注意边界 j > 0
  10. while (j > 0 && nums[j - 1] > temp) {
  11. nums[j] = nums[j - 1];
  12. j--;
  13. }
  14. nums[j] = temp;
  15. }
  16. return nums;
  17. }

优化:「将一个数字插入一个有序的数组」这一步,可以不使用逐步交换,使用先赋值给「临时变量」,然后「适当的元素」后移,空出一个位置,最后把「临时变量」赋值给这个空位的策略(就是上面那张图的意思)。编码的时候如果不小心,可能会把数组的值修改,建议多调试;
特点:「插入排序」可以提前终止内层循环(体现在 nums[j - 1] > temp 不满足时),在数组「几乎有序」的前提下,「插入排序」的时间复杂度可以达到 O(N);
由于「插入排序」在「几乎有序」的数组上表现良好,特别地,在「短数组」上的表现也很好。因为「短数组」的特点是:每个元素离它最终排定的位置都不会太远。为此,在小区间内执行排序任务的时候,可以转向使用「插入排序」。


几大基础排序
https://leetcode-cn.com/problems/sort-an-array/solution/fu-xi-ji-chu-pai-xu-suan-fa-java-by-liweiwei1419/