java常用排序算法

在各类算法中,排序算法是最基本的内容。在这里主要分享一个冒泡排序,选择排序,插入排序,希尔排序,快速排序和堆排序以及合并排序。

1、冒泡排序

这里是最基础的了,不用多说。

public static void bubbleSort(int[] a){
		int temp;
		for(int i=1;i<a.length;i++){
			for(int j=0;j<a.length-i;j++){
				if(a[j]>a[j+1]){
					temp=a[j];
					a[j]=a[j+1];
					a[j+1]=temp;
				}
			}
			System.out.print("第"+i+"步排序结果是:");
			for(int k=0;k<a.length;k++){
				System.out.print("\t"+a[k]);
			}
			System.out.print("\n");
		}
	}

2、选择排序

主要是通过选择和交换来实现排序,长得和冒泡差不多

public static void selectSort(int[] a){
		int index,temp;
		for(int i=0;i<a.length-1;i++){
			index=i;
			for(int j=i+1;j<a.length;j++){
				if(a[j]>a[index]){
					index=j;
				}
			}

			if(index!=i){
				temp=a[i];
				a[i]=a[index];
				a[index]=temp;
			}
			System.out.print("第"+i+"步排序结果是:");
			for(int k=0;k<a.length;k++){
				System.out.print("\t"+a[k]);
			}
			System.out.print("\n");

		}

	}

3、插入排序

主要是通过对未排序的数据执行逐个插入至合适的位置而完成排序工作。

static void insertionSort(int[] a ){
		int i,j,t,h;
		for(i=1;i<a.length;i++){
			t=a[i];
			j=i-1;
			while(j>=0 && t<a[j]){
				a[j+1]=a[j];
				j--;
			}
			a[j+1]=t;

			System.out.print("第"+i+"步排序结果是:");
			for(int k=0;k<a.length;k++){
				System.out.print("\t"+a[k]);
			}
			System.out.print("\n");
		}
	}

4、希尔排序

基于插入排序的思想,主要是将n个元素分成n/2个数字序列,第1个数据和第n/2+1个数据为一对,一次循环使每一个序列对排好顺序,然后再变为n/4个序列,再次排序,除的时候取整。

static void shellSort(int[] a ){
		int i,j,h;
		int r,temp;
		int x=0;

		for(r=a.length/2;r>=1;r/=2){

		for(i=r;i<a.length;i++){
			temp=a[i];
			j=i-r;
			while(j>=0 && temp<a[j]){
				a[j+r]=a[j];
				j-=r;
			}
			a[j+r]=temp;

		}
			x++;
			System.out.print("第"+x+"步排序结果是:");
			for(h=0;h<a.length;h++){
				System.out.print("\t"+a[h]);
			}
			System.out.print("\n");
		}

	}

5、快速排序

快速排序通过多次比较和交换来实现排序,首先设定一个分界值,通过该分界值的数据分为左右两部分,将大于等于分界值的数据集中到数组右边,小于的放到数组左边。然后左右两边的数据可以独立排序。重复上述过程。

static void quickSort(int[] arr,int left,int right){
		int f,t;
		int rtemp,ltemp;

		ltemp=left;
		rtemp=right;
		f=arr[(left+right)/2];

		while(ltemp<rtemp){
			while(arr[ltemp]<f){
				++ltemp;
			}
			while(arr[rtemp]>f){
				--rtemp;
			}
			if(ltemp<=rtemp){
				t=arr[ltemp];
				arr[ltemp]=arr[rtemp];
				arr[rtemp]=t;
				--rtemp;
				++ltemp;
			}
		}

		if(ltemp==rtemp){
			ltemp++;
		}
		if(left<rtemp){
			quickSort(arr,left,ltemp-1);
		}
		if(ltemp<right){
			quickSort(arr,rtemp+1,right);
		}
	}

6、堆排序

这个就稍稍有点复杂了哦。

一个完整的堆排序需要经过反复的两个步骤:构造堆结构和堆排序输出。

首先将原始的无序数据放置到一个完全二叉树的各个结点中,然后由完全二叉树的下层向上层逐层对父子结点的数据进行比较,

使父结点的数据大于子结点的数据。这里需要使用筛运算进行结点数据的调整,直到所有结点最后都满足堆结构的条件为止。

static void heapSort(int a[],int n){       //堆排序
		int i,j,h,k;
		int t;

		for(i=n/2-1;i>=0;i--){       //将a[0,n-1]建成大根堆
			while(2*i+1<n){        //第i个结点有右子树
				j=2*i+1;
				if((j+1)<n){
					if(a[j]<a[j+1])   //右左子树小于右子树,则需要比较右子树
						j++;     //序号增加1,指向右子树
				}
				if(a[i]<a[j]){     //比较i与j为序号的数据
					t=a[i];   //交换数据
					a[i]=a[j];
					a[j]=t;   //堆被破坏,需要重新调整
					i=j;
				}
				else{           //比较左右子树均大则堆未破坏,不再需要调整
					break;
				}
			}
		}
		System.out.println("原始构成的堆:");
		for(h=0;h<n;h++){
			System.out.print("\t"+a[h]);
		}
		System.out.print("\n");

		for(i=n-1;i>0;i--){
			t=a[0];              //与第i个记录交换
			a[0]=a[i];
			a[i]=t;
			k=0;

			while(2*k+1<i){      //第i个结点有右子树
				j=2*k+1;
				if((j+1)<i){
					if(a[j]<a[j+1]){    //右左子树小于右子树,则需要比较右子树
						j++;       //序号增加1,指向右子树
					}
				}
				if(a[k]<a[j]){          //比较i与j序号的数据
					t=a[k];            //交换数据
					a[k]=a[j];
					a[j]=t;
					k=j;              //堆被破坏,需要重新调整
				}else{
					break;
				}

			}

			System.out.println("第"+(n-i)+"步排序结果:");
			for(h=0;h<n;h++){
				System.out.print("\t"+a[h]);
			}
			System.out.print("\n");
		}

	}

	/**
	 * 参数a是以线性方式保存的二叉树数组,输入参数n为数组的长度。
	 *
	 * @param args
	 */

7、合并排序

一个待排序的原始数据排序进行合并排序的基本思路:首先将含有n个结点的待排序数据序列看成是由n个长度为1的有序子表组成,

* 将其依次两两合并,得到长度为2的若干有序子表;然后,再对

static void mergeOne(int a[],int b[],int n ,int len){   //完成一次合并的函数
		int i,j,k,s,e;
		s=0;
		while(s+len<n){
			e=s+2*len-1;
			if(e>=n){            //最后一段可能少于len个结点
				e=n-1;
			}
			//相邻有序段合并
			k=s;
			i=s;
			j=s+len;
			while(i<s+len && j<=e){   //如果两个有序表都未结束时,循环比较
				if(a[i]<=a[j]){      //如果较小的元素复制到数组b中
					b[k++]=a[i++];
				}else{
					b[k++]=a[j++];
				}
			}
			while(i<s+len){   //未合并的部分复制到数组b中
				b[k++]=a[i++];
			}
			while(j<=e){
				b[k++]=a[j++];   //未合并的部分复制到数组b中
			}
			s=e+1;                //下一对有序段中左段的开始下标

		}
		if(s<n){                 //将剩余的一个有序段从数组A复制到数组b中
			for(;s<n;s++){
				b[s]=a[s];
			}
		}
	}

		static void mergeSort(int a[],int n){  //合并排序
			int h,count,len,f;
			count=0;   //排序步骤
			len=1;    //有序序列长度
			f=0;       //变量f做标志

			int[] p=new int[n];
			while(len<n){
				if(f==1){            //交替在A和P之间合并
					mergeOne(p,a,n,len);  //p合并到a
				}else{
					mergeOne(a,p,n,len);    //a合并到p
				}
				len=len*2;                //增加有序序列长度
				f=1-f;                     //使f值在0和1之间切换

				count++;
				System.out.printf("第"+count+"排序结果:");

				for(h=0;h<SIZE;h++){
					System.out.print("\t"+a[h]);
				}
				System.out.print("\n");

			}
			if(f==1){             //如果进行了排序
				for(h=0;h<n;h++){   //将内存p中的数据复制回数组a
					a[h]=p[h];
				}
			}

		}

		/**
		 * 在MergeOne()方法中,输入参数a是一个数组,用来保存待排序的数据,输入参数b是一个数组,用来保存合并后的数据
		 * ,参数n表示数组a中需要进行排序的元素总数,参数len表示每个有序子表的长度。
		 *
		 * 二路合并排序算法需要申请较大的辅助内存空间,这个辅助空间的大小与待排序的原始序列一样多。
		 *
		 *
		 * @param args
		 */

在这里分享的集中排序算法中,冒泡排序,插入排序和合并排序都是稳定排序算法(即维持记录的相对次序来进行排序)。

没有一种排序算法是绝对好坏的,不同的排序算法各有优劣,在实际应用中药按实际情况来选择排序算法。如果数据量较小,可以采用插入排序或者选择排序,当数据量n较大时,则应该用时间复杂度为O(nlogn)的排序方法,如快速排序,堆排序或合并排序,用空间换时间。如果排序的原始数据呈随机分布,则用快速排序。

源码下载地址:点击打开链接http://download.csdn.net/detail/sdksdk0/9527723

时间: 2024-10-11 08:38:23

java常用排序算法的相关文章

Java常用排序算法+程序员必须掌握的8大排序算法+二分法查找法

Java 常用排序算法/程序员必须掌握的 8大排序算法 本文由网络资料整理转载而来,如有问题,欢迎指正! 分类: 1)插入排序(直接插入排序.希尔排序) 2)交换排序(冒泡排序.快速排序) 3)选择排序(直接选择排序.堆排序) 4)归并排序 5)分配排序(基数排序) 所需辅助空间最多:归并排序 所需辅助空间最少:堆排序 平均速度最快:快速排序 不稳定:快速排序,希尔排序,堆排序. 先来看看 8种排序之间的关系: 1.直接插入排序 (1)基本思想:在要排序的一组数中,假设前面(n-1)[n>=2]

Java 常用排序算法/程序员必须掌握的 8大排序算法

Java 常用排序算法/程序员必须掌握的 8大排序算法 分类: 1)插入排序(直接插入排序.希尔排序) 2)交换排序(冒泡排序.快速排序) 3)选择排序(直接选择排序.堆排序) 4)归并排序 5)分配排序(基数排序) 所需辅助空间最多:归并排序 所需辅助空间最少:堆排序 平均速度最快:快速排序 不稳定:快速排序,希尔排序,堆排序. 先来看看 8种排序之间的关系: 1.直接插入排序 (1)基本思想:在要排序的一组数中,假设前面(n-1)[n>=2] 个数已经是排 好顺序的,现在要把第n 个数插到前

Java常用排序算法/程序员必须掌握的8大排序算法

转载自http://blog.csdn.net/qy1387/article/details/7752973 分类: 1)插入排序(直接插入排序.希尔排序)2)交换排序(冒泡排序.快速排序)3)选择排序(直接选择排序.堆排序)4)归并排序5)分配排序(基数排序)所需辅助空间最多:归并排序所需辅助空间最少:堆排序平均速度最快:快速排序 不稳定:快速排序,希尔排序,堆排序.    [java] view plain copy print? // 排序原始数据 private static final

Java常用排序算法实现

目录 目录 概述 插入排序 希尔排序 简单排序 堆排序 冒泡排序 快速排序 源码 概述 这里总的给出了几种常用的排序算法,也是当作复习和巩固. 插入排序 希尔排序 简单排序 堆排序 冒泡排序 快速排序 以上几个都是根据排序的原理和原则自定义实现的.对于同一个排序实际的实现过程可能有一些不同,但思想和处理方式是相同的. 插入排序 基本思想:在要排序的一组数中,假设前面(n-1)[n>=2] 个数已经是排好顺序的,现在要把第n个数插到前面的有序数中,使得这n个数也是排好顺序的.如此反复循环,直到全部

[转]Java 常用排序算法/程序员必须掌握的 8大排序算法

本文转自:http://www.cnblogs.com/qqzy168/archive/2013/08/03/3219201.html 本文由网络资料整理转载而来,如有问题,欢迎指正! 分类: 1)插入排序(直接插入排序.希尔排序) 2)交换排序(冒泡排序.快速排序) 3)选择排序(直接选择排序.堆排序) 4)归并排序 5)分配排序(基数排序) 所需辅助空间最多:归并排序 所需辅助空间最少:堆排序 平均速度最快:快速排序 不稳定:快速排序,希尔排序,堆排序. 先来看看 8种排序之间的关系: 1.

Java常用排序

1 /** 2 * Java常用排序 3 * @author LiTaiQing 4 * 5 */ 6 public class SortTest { 7 8 /* 9 * 直接插入排序 10 * 基本思想: 11 * 在要排序的一组数中,假设前面(n-1)[n>=2] 个数已经是排好顺序的, 12 * 现在要把第n个数插到前面的有序数中,使得这n个数也是排好顺序的. 13 * 如此反复循环,直到全部排好顺序 14 */ 15 public static void insertSort(int[

常用排序算法比较与分析

一.常用排序算法简述 下面主要从排序算法的基本概念.原理出发,分别从算法的时间复杂度.空间复杂度.算法的稳定性和速度等方面进行分析比较.依据待排序的问题大小(记录数量 n)的不同,排序过程中需要的存储器空间也不同,由此将排序算法分为两大类:[内排序].[外排序]. 内排序:指排序时数据元素全部存放在计算机的随机存储器RAM中. 外排序:待排序记录的数量很大,以致内存一次不能容纳全部记录,在排序过程中还需要对外存进行访问的排序过程. 先了解一下常见排序算法的分类关系(见图1-1) 图1-1 常见排

java希尔排序算法

原文:java希尔排序算法 代码下载地址:http://www.zuidaima.com/share/1550463279090688.htm 希尔排序算法的基本思想是:先取一个小于n的整数d1作为第一个增量,把文件的全部记录分成d1个组.所有距离为dl的倍数的记录放在同一个组中.先在各组内进行直接插人排序:然后,取第二个增量d2<d1重复上述的分组和排序,直至所取的增量dt=1(dt<dt-l<-<d2<d1),即所有记录放在同一组中进行直接插入排序为止.该方法实质上是一种

七种常用排序算法

七种常用排序算法 一.常见排序算法一览: 时间复杂度: 是一个函数,它定量描述了该算法的运行时间. 空间复杂度:一个算法在运行过程中临时占用存储空间大小的量度. 稳定性:保证排序前2个相等的数其在序列的前后位置顺序和排序后它们两个的前后位置顺序相同就稳定,反之不稳定. 视觉直观感受 7 种常用的排序算法 二.算法C#实现: 1. 直接插入排序: using System; using System.Collections.Generic; using System.Linq; using Sys