快速、归并、计数排序算法(Java)

本文详细介绍了快速排序、归并排序和计数排序的基本思想、算法步骤、时间与空间复杂度,并提供了Java代码实现。通过分析,指出在不同场景下选择不同排序算法的重要性,例如快速排序的空间效率、归并排序的稳定性以及计数排序在特定条件下的高效性。
摘要由CSDN通过智能技术生成

快速排序算法(Quick Sort)

基本思想

通过一趟排序将待排记录分隔成独立的两部分,其中一部分记录的关键字均比另一部分的关键字小,则可分别对这两部分记录继续进行排序,以达到整个序列有序。

算法步骤

1、设置第一个元素值为基准值key,将数列中比key小的数放到key左边并重新组成一个新数列r_left,比key大的放到其右边重新组成一个数列r_right。

2、重复第一个步骤,递归地对子序列r_left和r_right进行划分,直至所有数列都为空数列,排序结束。

举例说明

对 r [ ] ={49,38,65,97,76,13,27,49}进行从小到大排序。

在这里插入图片描述

1、令key = r[0] = 49,并设置左右指针 i 和 j ,分别指向数列的最左端和最右端;

2、指针 j 向左滑动,直至 r[j] < key,则令 r[i] = r[j],且 i++;

3、指针 i 向右滑动,直至 r[i] > key,则令 r[j] = r[i],且 j - -;

4、重复步骤2、3,直至 i = j,则令 r[i] = key,第一次划分结束;

  第一次划分的结果:[27 38 13] 49 [76 97 65 49]

  子序列 r_left :27 38 13

  子序列 r_right :76 97 65 49

5、对子序列重复步骤1~4,继续划分子序列,直到所有子序列为空序列,则排序结束。

  第二次划分的结果:[13] 27 [38] 49 [49 65] 76 [97]

  第三次划分的结果:13 27 38 49 49 [65] 76 97

  第四次划分的结果:13 27 38 49 49 65 76 97  //排序结束

时间与空间复杂度

  • 时间复杂度:

    • 最好情况:O(nlogn)
    • 最坏情况:O(n^2)
    • 平均情况:O(nlogn)
  • 空间复杂度:O(logn)

快排算法的空间复杂度是固定的,但时间复杂度根据实际情况而变。因此,在实际情况中,我们要先分析要进行排序的数列的特征,再选择合适的排序算法。

最好的情况:每次划分所选择的中间数恰好将当前序列几乎等分,经过 logn 趟划分,便可得到长度为1的子表。

最坏情况:每次所选的中间数是当前序列中的最大或最小元素,这使得每次划分所得的子表中一个为空表,另一子表的长度为原表的长度 -1。

归并排序算法(Merge Sort)

基本思想

将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列分为若干个子序列,每个子序列是有序的。然后再把有序子序列合并为整体有序序列。

算法步骤

1、先将数列一直对半分成多个子数列,直到所有子数列都为有序序列或者子数列只有一个元素;

2、对所有的子序列进行并归。

时间与空间复杂度

  • 时间复杂度:O(nlogn)
  • 空间复杂度:O(n)

归并排序的时间复杂度,在前面中提到的最坏,最好和平均情况下都是O(nlogn)(与快速排序的最好情况相同),这说明它性能非常好的排序算法。

但是它需要占用 O(n)的内存空间,如果数据量一旦很大,内存可能吃不消。而快速排序算法是就地排序算法,不占用额外的内存空间。针对占用内存问题,目前好像有方法已经可以将归并排序算法改进为就地排序算法,大家可以先自行了解一下。

计数排序算法(Counting Sort)

基本思想

对于给定的输入序列中的每一个元素x,确定该序列中值小于x的元素的个数(此处并非比较各元素的大小,而是通过对元素值的计数和计数值的累加来确定)。一旦有了这个信息,就可以将x直接存放到最终的输出序列的正确位置上。

算法步骤

1、找出数列中的最小值min与最大值max,确定数值区间 [min,max] 并求出数值组长度 k=max-min+1;

如{9,4,4,1}:min=1,max=9,区间 [1,9],长度 k=9

2、标记各个数值在要排列数列中的出现次数;

[1,9] 每个数值出现的次数 d[ ] ={1,0,0,2,0,0,0,0,1}

3、根据每个数值 i 出现的次数d[i],依次输出d[i]个相应数值,即可得到排序结果。

最后输出排序结果:1 4 4 9

时间与空间复杂度

  • 时间复杂度:O(n+k)
  • 空间复杂度:O(n+k)

其中 k=max-min+1 为区间长度。

由于计数排序是一种非比较排序算法,它的优势在于在对一定范围内的整数排序时,快于任何比较排序算法(如前面的快速排序与归并排序)。当k不是很大并且序列比较集中时,计数排序是一个很有效的排序算法。

如果 k 很大,则需要申请大量的空间来存储数值区间内每个数值在数列中出现的次数d[ ]。在这种情况下,不适合使用计数排序算法。

java代码

话不多说,直接给出代码。

public class quickSort {
   
	public static void main(String[] args) {
   
		long startTime = System.nanoTime();//测时间:currentTimeMillis表示毫秒,nanoTime表示纳秒
		int[] arr= {
   6, 9, 1, 4, 5, 8, 7, 0, 2, 3, 100};
		int right=arr.length-1;
		quickSort(arr, 0, right);
		for (int i = 0; i < arr.length; i++) 
  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值