排序算法:归并排序、快速排序

排序算法:归并排序、快速排序

 

相关博客:

排序算法:冒泡排序、插入排序、选择排序、希尔排序

排序算法:归并排序、快速排序

排序算法:桶排序、计数排序、基数排序

排序算法:堆排序

十大排序算法小结


一、归并排序:

1、工作原理:

归并排序的采用分治思想,如果要排序一个数组,我们先把数组从中间分成前后两个部分,然后对前后两个部分分别进行排序,再将排好序的两部分合并在一起,这样整个数组就都有序了。

2、动图演示:

排序算法:归并排序、快速排序

3、Java代码实现:

public class MergeSort {
	//归并排序:
	public void mergeSort(int[] a,int p,int r){
		//递归终止条件
		if(p>=r) return;
		
		int q=(p+r)/2;//取数组的中间点
		mergeSort(a,p,q);//对前半部分进行排序
		mergeSort(a,q+1,r);//对后半部分进行排序
		
		merge(a, p, q, r);//合并前后两个部分
	}
	
        //merge()函数是归并排序的重点
	public void merge(int[] a,int p,int q,int r){
		//初始化变量:i表示前半部分的下标,j表示后半部分的下标,k表示临时数组的下标
		int i=p;int j=q+1;int k=0;
		int[] tmp = new int[r-p+1];
		
		while(i<=q && j<=r){
			if(a[i]<=a[j]){
				tmp[k++]=a[i++];
			}else{
				tmp[k++]=a[j++];
			}
		}
		
		//将剩余数据拷贝到临时数组中
		for(;i<=q;i++) tmp[k++]=a[i];
		for(;j<=r;j++) tmp[k++]=a[j];
		
		//将tmp临时数组拷贝回原数组
		for(int x=0;x<r-p+1;x++) a[p+x]=tmp[x];
	}
}

4、算法分析:

(1)归并排序是一种稳定的排序算法;

(2)最好、最好、平均时间复杂度都是O(nlogn)。

(3)空间复杂度是O(n),所以不是原地排序。这也是归并排序的一个致命弱点。


二、快速排序:

1、工作原理:

快速排序也是利用分治思想。如果要排序一组数据,我们先选择这组数据中任意一个数据作为分区点pivot,然后遍历这组数据,将小于分区点pivot的放到左边,大于分区点pivot的放到右边,将pivot放到中间。然后再分别对左右两部分进行排序。

2、图片演示:

排序算法:归并排序、快速排序

3、Java代码实现:

public class QuickSort {

	public void quickSort(int[] a,int p,int r){
		//递归终止条件:
		if(p>=r) return;
		
		//获取分区点:
		int q=partition(a,p,r);
		quickSort(a,p,q-1);//对左分区进行排序
		quickSort(a,q+1,r);//对右分区进行排序
	}
	
	//返回分区点,快速排序的核心:
	public int partition(int[] a,int p,int r){
		
		int i=p;int pivot = a[r];
		for(int j=p;j<=r;j++){
			if(a[j]<pivot){
				swap(a,i,j);
				i++;
			}
		}
		swap(a, i, r);
		return i;
	}
	
	public void swap(int[] a,int x,int y){
		int temp=a[x];
		a[x]=a[y];
		a[y]=temp;
	}
}

其中,快速排序的核心就是paitition()分区函数,它的过程如下图所示:

排序算法:归并排序、快速排序

4、算法分析:

(1)快速排序是一种原地排序,空间复杂度为O(1)。

(2)快速排序是一种不稳定算法。

(3)快速排序是的最好时间复杂度、平均时间复杂度为O(nlogn)。但是在极端情况下,如果数组中的数据原来就是有序的,比如1,3,5,6,8,如果我们每次选择最后一个元素作为pivot,那每次分区得到的两个区间就是不均等的,我们需要大约进行n次分区操作,才能完成快排的整个过程,每次分区我们平均要扫描大约n/2个元素,这时时间复杂度为O(n^2),也就是最坏时间复杂度。


三、小结:归并与快排的区别:

(1)归并排序和快速排序都是用分治的思想,代码都是通过递归来实现的,但是归并排序的核心是merge()合并函数,而快速排序的核心是partition()分区函数。归并排序的处理过程是从下到上,先处理子问题,然后再合并。而快速排序的处理过程正好相反,它的处理过程是从上到下的,先分区,然后再处理子问题。这两种排序的处理过程如下图所示:

排序算法:归并排序、快速排序

(2)归并排序算法是一种任何情况下时间复杂度都比较稳定的排序算法,时间复杂度都是O(nlogn);快速排序算法最坏情况下时间复杂度为O(n^2),但是平均时间复杂度都是O(nlogn),不仅如此,快速排序时间复杂度退化到O(n^2)的概率非常小,我们可以通过合理选择pivot来避免这种情况。

(3)归并排序不是原地排序算法,空间复杂度比较高,为O(n);快速排序是原地排序算法,空间复杂度为O(1)。

(4)归并排序是稳定的算法,快速排序不稳定。

 

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/114619.html原文链接:https://javaforall.net

(0)
上一篇 2021年10月5日 上午7:00
下一篇 2021年10月5日 上午8:00


相关推荐

  • js定时跳转网页_js 网页代码

    js定时跳转网页_js 网页代码效果如下:五秒跳完之后,转到百度的页面js代码如下:window.οnlοad=init;functioninit(){window.setTimeout(“tiaozhuan()”,5000);window.setInterval(“shijian()”,1000);//五秒后调用tiaozhuan}functiontiaozhuan(){location.replace(“http://www.baidu.com”);} functionshijian(){ var

    2022年8月12日
    11
  • 老程序员Java数组转List都这样操作「建议收藏」

    老程序员Java数组转List都这样操作「建议收藏」一、使用Arrays.asList()通过Arrays.asList(strArray)方式,将数组转换List后,不能对List增删,只能查改,否则抛异常。测试代码如下:publicstaticvoidway1(){List<String>list=Arrays.asList(“1″,”2”);//对转换后的list插入一条数据list.add(“3”);System.out.println(lis

    2022年8月23日
    7
  • 图像分割的原则_常用的图像分割方法

    图像分割的原则_常用的图像分割方法在对处理后的图像数据进行分析之前,图像分割是最重要的步骤之一。它的主要目标是将图像化分为与其中含有的真实世界的物体或区域有枪相关性的组成部分。根据目标可将图像分割分为:1.完全分割——结果是

    2022年8月1日
    3
  • java卸载dll,如何在java中卸载Dll?[通俗易懂]

    java卸载dll,如何在java中卸载Dll?[通俗易懂]IhavewrittenaJavaagentinLotusNotes8.5toresettoken’sPINusingIAIKPKCS11wrapper.Whentheagentisloadedforthefirsttimeitworkedfine.ButafterIcloseandrestarttheagentthenitf…

    2022年5月19日
    82
  • HTTP状态码——302「建议收藏」

    理解302表示临时性重定向。访问一个Url时,被重定向到另一个url上。常用于页面跳转。与301的区别301是指永久性的移动,302是暂时性的,即以后还可能有变化其它重定向方式在响应头中加入Location参数。浏览器接受到带有location头的响应时,就会跳转到相应的地址。…

    2022年4月18日
    90
  • DateAdd 函数

    DateAdd 函数DateAdd 函数返回已添加指定时间间隔的日期 DateAdd interval number date 参数 interval 必选项 字符串表达式 表示要添加的时间间隔 有关数值 请参阅 设置 部分 number 必选项 数值表达式 表示要添加的时间间隔的个数 数值表达式可以是正数 得到未来的日期 或负数 得到过去的日期 date

    2026年3月16日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号