zoukankan      html  css  js  c++  java
  • 【LeetCode解题总结】排序篇

    基本的排序算法

    冒泡排序和插入排序是最基础的,面试官有时候喜欢拿它们来考察你的基础知识,并且看看你能不能快速地写出没有 bug 的代码。

    冒泡排序(Bubble Sort)

    实现

    每一轮,从杂乱无章的数组头部开始,每两个元素比较大小并进行交换,直到这一轮当中最大或最小的元素被放置在数组的尾部,然后不断地重复这个过程,直到所有元素都排好位置。其中,核心操作就是元素相互比较。

    复杂度

    空间复杂度 O(1)
    平均时间复杂度 O(n^2)
    稳定的排序算法

    插入排序(Insertion Sort)

    实现

    不断地将尚未排好序的数插入到已经排好序的部分。

    复杂度

    空间复杂度O(1)
    平均时间复杂度O(n^2)
    稳定的排序算法

    高级的排序算法

    归并排序、快速排序和拓扑排序的思想是解决绝大部分涉及排序问题的关键

    归并排序

    实现

    一开始先把数组从中间划分成两个子数组,一直递归地把子数组划分成更小的子数组,直到子数组里面只有一个元素,才开始排序。

    void sort(int[] A, int lo, int hi) {
    	  // 判断是否只剩下最后一个元素
    	  if (lo >= hi) return;
    	  // 从中间将数组分成两个部分
    	  int mid = lo + (hi - lo) / 2;
    	  // 分别递归地将左右两半排好序
    	  sort(A, lo, mid);
    	  sort(A, mid + 1, hi);
    	  // 将排好序的左右两半合并  
    	  merge(A, lo, mid, hi);
    	}
    void merge(int[] nums, int lo, int mid, int hi) {
        // 复制一份原来的数组
        int[] copy = nums.clone();
        // 定义一个 k 指针表示从什么位置开始修改原来的数组
      	// i 指针表示左半边的起始位置,j 表示右半边的起始位置
        int k = lo, i = lo, j = mid + 1;
        while (k <= hi) {
            if (i > mid) {
                nums[k++] = copy[j++];
            } else if (j > hi) {
              nums[k++] = copy[i++];
            } else if (copy[j] < copy[i]) {
              nums[k++] = copy[j++];
            } else {
              nums[k++] = copy[i++];
            }
        }
    }
    
    
    

    复杂度

    时间复杂度O(nlogn)
    对于规模为 n 的问题,一共要进行 log(n) 层的大小切分。在每一层里,我们都要进行合并,所涉及到的元素其实就是数组里的所有元素,因此,每一层的合并复杂度都是 O(n),所以整体的复杂度就是O(nlogn)

    空间复杂度 O(n)

    由于合并 n 个元素需要分配一个大小为 n 的额外数组,合并完成之后,这个数组的空间就会被释放,所以算法的空间复杂度就是 O(n)

    归并排序也是稳定的排序算法

    总结

    归并算法的思想很重要,其中对两个有序数组合并的操作,在很多面试题里都有用到,建议大家一定要把这个算法练熟。

    快速排序(Quick Sort)

    实现

    利用基准值来将原数组分为两个子数组,然后分别对每个子数组排序

    void sort(int[] nums, int lo, int hi) {
    	    if (lo >= hi) return; // 判断是否只剩下一个元素,是,则直接返回
    	    // 利用 partition 函数找到一个随机的基准点
    	    int p = partition(nums, lo, hi);
    	    // 递归地对基准点左半边和右半边的数进行排序
    	    sort(nums, lo, p - 1);
    	    sort(nums, p + 1, hi);
    	}
    int partition(int[] nums, int lo, int hi) {
    	    // 随机选择一个数作为基准值,nums[hi] 就是基准值
    	    swap(nums, randRange(lo, hi), hi);
    	    int i, j;
    	    // 从左到右用每个数和基准值比较,若比基准值小,则放到指针 i 所指向的位置
    	    // 循环完毕后,i 指针之前的数都比基准值小
    	    for (i = lo, j = lo; j < hi; j++) {
    	        if (nums[j] <= nums[hi]) {
    	            swap(nums, i++, j);
    	        }
    	    }
    	    // 末尾的基准值放置到指针 i 的位置,i 指针之后的数都比基准值大
    	    swap(nums, i, j);
    	    // 返回指针 i,作为基准点的位置
    	    return i;
    }
    

    复杂度

    时间复杂度

    最优情况:被选出来的基准值都是当前子数组的中间数。O(nlogn)
    最坏情况:基准值选择了子数组里的最大或者最小值。O(n^2)

    空间复杂度
    和归并排序不同,快速排序在每次递归的过程中,只需要开辟 O(1) 的存储空间来完成交换操作实现直接对数组的修改,又因为递归次数为 logn,所以它的整体空间复杂度完全取决于压堆栈的次数,因此它的空间复杂度是 O(logn)。

    不稳定排序

    解法总结

    1. 快排拓展

      题目 思路
      215. 数组中的第K个最大元素
  • 相关阅读:
    linux
    JAVA——遍历
    linux基础命令
    Java的反射和代理以及注解
    web 前端遇到的问题
    JTopo 使用
    阿里云服务器linux(cenos)下 jdk、tomcat的安装配置
    笔记--数据库总结
    0.01的区别
    犹豫——辛苦——坚持——收获 (2019北航软工培训总结)
  • 原文地址:https://www.cnblogs.com/JasonBUPT/p/11868406.html
Copyright © 2011-2022 走看看