数据结构与算法之排序(冒泡、选择、插入、希尔、归并、快速)(三)

简介: 数据结构与算法之排序(冒泡、选择、插入、希尔、归并、快速)

2.3.快速排序


快速排序是对冒泡排序的一种改进。它的基本思想是:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。


需求:

排序前:{6, 1, 2, 7, 9, 3, 4, 5, 8}

排序后:{1, 2, 3, 4, 5, 6, 7, 8, 9}

排序原理:


1.首先设定一个分界值,通过该分界值将数组分成左右两部分;


2.将大于或等于分界值的数据放到到数组右边,小于分界值的数据放到数组的左边。此时左边部分中各元素都小于或等于分界值,而右边部分中各元素都大于或等于分界值;


3.然后,左边和右边的数据可以独立排序。对于左侧的数组数据,又可以取一个分界值,将该部分数据分成左右两部分,同样在左边放置较小值,右边放置较大值。右侧的数组数据也可以做类似处理。


4.重复上述过程,可以看出,这是一个递归定义。通过递归将左侧部分排好序后,再递归排好右侧部分的顺序。当左侧和右侧两个部分的数据排完序后,整个数组的排序也就完成了


81fa904199294dbda4f114906cd4e1be.png


快速排序 API设计:


image.png


切分原理:


把一个数组切分成两个子数组的基本思想:


1.找一个基准值,用两个指针分别指向数组的头部和尾部;


2.先从尾部向头部开始搜索一个比基准值小的元素,搜索到即停止,并记录指针的位置;


3.再从头部向尾部开始搜索一个比基准值大的元素,搜索到即停止,并记录指针的位置;


4.交换当前左边指针位置和右边指针位置的元素;


5.重复2,3,4步骤,直到左边指针的值大于右边指针的值停止。


6d2c9e4c692d4e4f8fa6fe09f3558195.png

32d4826a70a342c385bf40697c709690.png


快速排序代码实现:


// 排序代码
public class Quick {
  public static void sort(Comparable[] a) {
    int lo = 0;
    int hi = a.length - 1;
    sort(a, lo,hi);
  }
  private static void sort(Comparable[] a, int lo, int hi) {
    if (hi<=lo){
      return;
    }
  //对a数组中,从lo到hi的元素进行切分
  int partition = partition(a, lo, hi);
  //对左边分组中的元素进行排序
  //对右边分组中的元素进行排序
  sort(a,lo,partition-1);
  sort(a,partition+1,hi);
  }
  public static int partition(Comparable[] a, int lo, int hi) {
    Comparable key=a[lo];//把最左边的元素当做基准值
    int left=lo;//定义一个左侧指针,初始指向最左边的元素
    int right=hi+1;//定义一个右侧指针,初始指向左右侧的元素下一个位置
    //进行切分
    while(true){
      //先从右往左扫描,找到一个比基准值小的元素
      while(less(key,a[--right])){//循环停止,证明找到了一个比基准值小的元素
        if (right==lo){
        break;//已经扫描到最左边了,无需继续扫描
      }
    }
    //再从左往右扫描,找一个比基准值大的元素
    while(less(a[++left],key)){//循环停止,证明找到了一个比基准值大的元素
    if (left==hi){
      break;//已经扫描到了最右边了,无需继续扫描
    }
  }
  if (left>=right){
    //扫描完了所有元素,结束循环
    break;
  }else{
    //交换left和right索引处的元素
    exch(a,left,right);
  }
}
  //交换最后rigth索引处和基准值所在的索引处的值
  exch(a,lo,right);
  return right;//right就是切分的界限
}
  /*
  数组元素i和j交换位置
  */
  private static void exch(Comparable[] a, int i, int j) {
    Comparable t = a[i];
    a[i] = a[j];
    a[j] = t;
  }
  /*
  比较v元素是否小于w元素
  */
  private static boolean less(Comparable v, Comparable w) {
    return v.compareTo(w) < 0;
  }
}
//测试代码
public class Test {
  public static void main(String[] args) throws Exception {
    Integer[] arr = {6, 1, 2, 7, 9, 3, 4, 5, 8};
    Quick.sort(arr);
    System.out.println(Arrays.toString(arr));
  }
}


快速排序和归并排序的区别:


快速排序是另外一种分治的排序算法,它将一个数组分成两个子数组,将两部分独立的排序。快速排序和归并排序是互补的:归并排序将数组分成两个子数组分别排序,并将有序的子数组归并从而将整个数组排序,而快速排序的方式则是当两个数组都有序时,整个数组自然就有序了。在归并排序中,一个数组被等分为两半,归并调用发生在处理整个数组之前,在快速排序中,切分数组的位置取决于数组的内容,递归调用发生在处理整个数组之后。


快速排序时间复杂度分析:


快速排序的一次切分从两头开始交替搜索,直到left和right重合,因此,一次切分算法的时间复杂度为O(n),但整个快速排序的时间复杂度和切分的次数相关。最优情况:每一次切分选择的基准数字刚好将当前序列等分


b2e3152b6be04fc3b71305f635f1e90b.png


如果我们把数组的切分看做是一个树,那么上图就是它的最优情况的图示,共切分了 logn次,所以,最优情况下快速排序的时间复杂度为O(nlogn);


最坏情况:每一次切分选择的基准数字是当前序列中最大数或者最小数,这使得每次切分都会有一个子组,那么总共就得切分n次,所以,最坏情况下,快速排序的时间复杂度为O(n^2);


36ca7d43101b4a4280f46003997a0453.png

平均情况:每一次切分选择的基准数字不是最大值和最小值,也不是中值,这种情况我们也可以用数学归纳法证明,快速排序的时间复杂度为O(nlogn),由于数学归纳法有很多数学相关的知识,容易使我们混乱,所以这里就不对平均情况的时间复杂度做证明了


2.4.排序的稳定性


数组arr中有若干元素,其中A元素和B元素相等,并且A元素在B元素前面,如果使用某种排序算法排序后,能够保证A元素依然在B元素的前面,可以说这个该算法是稳定的


1572d6ca1fa54ad3b5c5754e32dc9e4b.png

稳定性的意义:


如果一组数据只需要一次排序,则稳定性一般是没有意义的,如果一组数据需要多次排序,稳定性是有意义的。例如要排序的内容是一组商品对象,第一次排序按照价格由低到高排序,第二次排序按照销量由高到低排序,如果第二次排序使用稳定性算法,就可以使得相同销量的对象依旧保持着价格高低的顺序展现,只有销量不同的对象才需要重新排序。这样既可以保持第一次排序的原有意义,而且可以减少系统开销


第一次按照价格从低到高排序:


image.png


第二次按照销量进行从高到低排序:


image.png


常见排序算法的稳定性:


冒泡排序:


只有当arr[i]>arr[i+1]的时候,才会交换元素的位置,而相等的时候并不交换位置,所以冒泡排序是一种稳定排序算法。


选择排序:


选择排序是给每个位置选择当前元素最小的,例如有数据{5(1),8 ,5(2), 2, 9 },第一遍选择到的最小元素为2,所以5(1)会和2进行交换位置,此时5(1)到了5(2)后面,破坏了稳定性,所以选择排序是一种不稳定的排序算法。


插入排序:


比较是从有序序列的末尾开始,也就是想要插入的元素和已经有序的最大者开始比起,如果比它大则直接插入在其后面,否则一直往前找直到找到它该插入的位置。如果碰见一个和插入元素相等的,那么把要插入的元素放在相等元素的后面。所以,相等元素的前后顺序没有改变,从原无序序列出去的顺序就是排好序后的顺序,所以插入排序是稳定的。


希尔排序:


希尔排序是按照不同步长对元素进行插入排序 ,虽然一次插入排序是稳定的,不会改变相同元素的相对顺序,但在不同的插入排序过程中,相同的元素可能在各自的插入排序中移动,最后其稳定性就会被打乱,所以希尔排序是不稳定的。


归并排序:


归并排序在归并的过程中,只有arr[i]<arr[i+1]的时候才会交换位置,如果两个元素相等则不会交换位置,所以它并不会破坏稳定性,归并排序是稳定的。


快速排序:


快速排序需要一个基准值,在基准值的右侧找一个比基准值小的元素,在基准值的左侧找一个比基准值大的元素,然后交换这两个元素,此时会破坏稳定性,所以快速排序是一种不稳定的算法。






目录
相关文章
|
2月前
|
搜索推荐 算法 C语言
【排序算法】八大排序(下)(c语言实现)(附源码)
本文继续学习并实现了八大排序算法中的后四种:堆排序、快速排序、归并排序和计数排序。详细介绍了每种排序算法的原理、步骤和代码实现,并通过测试数据展示了它们的性能表现。堆排序利用堆的特性进行排序,快速排序通过递归和多种划分方法实现高效排序,归并排序通过分治法将问题分解后再合并,计数排序则通过统计每个元素的出现次数实现非比较排序。最后,文章还对比了这些排序算法在处理一百万个整形数据时的运行时间,帮助读者了解不同算法的优劣。
142 7
|
2月前
|
搜索推荐 算法 C语言
【排序算法】八大排序(上)(c语言实现)(附源码)
本文介绍了四种常见的排序算法:冒泡排序、选择排序、插入排序和希尔排序。通过具体的代码实现和测试数据,详细解释了每种算法的工作原理和性能特点。冒泡排序通过不断交换相邻元素来排序,选择排序通过选择最小元素进行交换,插入排序通过逐步插入元素到已排序部分,而希尔排序则是插入排序的改进版,通过预排序使数据更接近有序,从而提高效率。文章最后总结了这四种算法的空间和时间复杂度,以及它们的稳定性。
121 8
|
3月前
|
搜索推荐 Shell
解析排序算法:十大排序方法的工作原理与性能比较
解析排序算法:十大排序方法的工作原理与性能比较
89 9
|
3月前
|
算法 搜索推荐 Java
数据结构与算法学习十三:基数排序,以空间换时间的稳定式排序,速度很快。
基数排序是一种稳定的排序算法,通过将数字按位数切割并分配到不同的桶中,以空间换时间的方式实现快速排序,但占用内存较大,不适合含有负数的数组。
42 0
数据结构与算法学习十三:基数排序,以空间换时间的稳定式排序,速度很快。
|
3月前
|
存储 搜索推荐 算法
【用Java学习数据结构系列】七大排序要悄咪咪的学(直接插入,希尔,归并,选择,堆排,冒泡,快排)以及计数排序(非比较排序)
【用Java学习数据结构系列】七大排序要悄咪咪的学(直接插入,希尔,归并,选择,堆排,冒泡,快排)以及计数排序(非比较排序)
34 1
|
3月前
|
数据可视化 搜索推荐 Python
Leecode 刷题笔记之可视化六大排序算法:冒泡、快速、归并、插入、选择、桶排序
这篇文章是关于LeetCode刷题笔记,主要介绍了六大排序算法(冒泡、快速、归并、插入、选择、桶排序)的Python实现及其可视化过程。
25 0
|
3月前
|
算法
❤️算法笔记❤️-(每日一刷-83、删除排序链表中的重复项)
❤️算法笔记❤️-(每日一刷-83、删除排序链表中的重复项)
36 0
|
3月前
|
搜索推荐 算法
排序算法---冒泡&选择&插入总结
排序算法---冒泡&选择&插入总结
21 0
|
3月前
|
存储 算法 搜索推荐
算法进阶之路:Python 归并排序深度剖析,让数据排序变得艺术起来!
算法进阶之路:Python 归并排序深度剖析,让数据排序变得艺术起来!
84 0
|
5月前
|
算法 搜索推荐 Java
算法实战:手写归并排序,让复杂排序变简单!
归并排序是一种基于“分治法”的经典算法,通过递归分割和合并数组,实现O(n log n)的高效排序。本文将通过Java手写代码,详细讲解归并排序的原理及实现,帮助你快速掌握这一实用算法。
51 0