`

java实现排序算法之插入排序(直接插入排序、折半插入、shell排序)

阅读更多
插入排序主要包括直接插入排序、shell排序和折半插入等几种排序。这篇文章主要说明直接插入排序、shell排序和折半插入三种排序的java实现。
一、直接插入排序
 直接插入排序(straight insertion sort)的作法是:
  每次从无序表中取出第一个元素,把它插入到有序表的合适位置,使有序表仍然有序。
  第一趟比较前两个数,然后把第二个数按大小插入到有序表中; 第二趟把第三个数据与前两个数从前向后扫描,把第三个数按大小插入到有序表中;依次进行下去,进行了(n-1)趟扫描以后就完成了整个排序过程。
  直接插入排序属于稳定的排序,时间复杂性为o(n^2),空间复杂度为O(1)。
  直接插入排序是由两层嵌套循环组成的。外层循环标识并决定待比较的数值。内层循环为待比较数值确定其最终位置。直接插入排序是将待比较的数值与它的前一个数值进行比较,所以外层循环是从第二个数值开始的。当前一数值比待比较数值大的情况下继续循环比较,直到找到比待比较数值小的并将待比较数值置入其后一位置,结束该次循环。
  值得注意的是,我们必需用一个存储空间来保存当前待比较的数值,因为当一趟比较完成时,我们要将待比较数值置入比它小的数值的后一位 插入排序类似玩牌时整理手中纸牌的过程。插入排序的基本方法是:每步将一个待排序的记录按其关键字的大小插到前面已经排序的序列中的适当位置,直到全部记录插入完毕为止。
//定义一个数据包装类
public class DataWrap implements Comparable<DataWrap>  
{
	int data;
	String flag;
	public DataWrap(int data, String flag)
	{
		this.data = data;
		this.flag = flag;
	}
	public String toString()
	{
		return data + flag;
	}
	//根据data实例变量来决定两个DataWrap的大小
	public int compareTo(DataWrap dw)
	{
		return this.data > dw.data ? 1 
			: (this.data == dw.data ? 0 : -1);
	}
}


public class InsertSort
{
	public static void insertSort(DataWrap[] data) 
	{
		System.out.println("开始排序:\n");
		int arrayLength = data.length;
		for (int i = 1 ; i < arrayLength ; i++ )
		{
			//当整体后移时,保证data[i]的值不会丢失
			DataWrap tmp = data[i];
			//i索引处的值已经比前面所有值都大,表明已经有序,无需插入
			//(i-1索引之前的数据已经有序的,i-1索引处元素的值就是最大值)
			if (data[i].compareTo(data[i - 1]) < 0)
			{
				int j = i - 1;
				//整体后移一格
				for ( ; j >= 0 && data[j].compareTo(tmp) > 0 ; j--)
				{
					data[j + 1] = data[j];
				}
				//最后将tmp的值插入合适位置
				data[j + 1] = tmp;
			}
			System.out.println(java.util.Arrays.toString(data));
		}
	}
	public static void main(String[] args)
	{
		DataWrap[] data = {
			new DataWrap(9 , ""),
			new DataWrap(-16 , ""),
			new DataWrap(21 , "*"),
			new DataWrap(23 , ""),
			new DataWrap(-30 , ""),
			new DataWrap(-49 , ""),
			new DataWrap(21 , ""),
			new DataWrap(30 , "*"),
			new DataWrap(30 , "")
		};
		System.out.println("排序之前:\n"
			+ java.util.Arrays.toString(data));
		insertSort(data);
		System.out.println("排序之后:\n" 
			+ java.util.Arrays.toString(data));
	}
}

二、折半插入排序
折半插入排序的基本思想是:设在顺序表中有一个对象序列V[0],V[1],…,V[n-1]。其中,V[0],V[1],…,V[i-1]是已经排好序的对象。在插入V[i]时,利用折半查找法寻找V[i]的插入位置。
public class BinaryInsertSort
{
	public static void binaryInsertSort(DataWrap[] data)
	{
		System.out.println("开始排序:\n");
		int arrayLength = data.length;
		for(int i = 1 ; i < arrayLength ; i++)
		{
			//当整体后移时,保证data[i]的值不会丢失
			DataWrap tmp = data[i];
			int low = 0;
			int high = i - 1;
			while(low <= high)
			{
				//找出low、high中间的索引
				int mid = (low + high) / 2;
				//如果tmp值大于low、high中间元素的值
				if(tmp.compareTo(data[mid]) > 0)
				{
					//限制在索引大于mid的那一半中搜索
					low = mid + 1;
				} 
				else
				{
					//限制在索引小于mid的那一半中搜索
					high = mid - 1;
				}
			}
			//将low到i处的所有元素向后整体移一位
			for(int j = i ; j > low ; j--)
			{
				data[j] = data[j - 1];
			}
			//最后将tmp的值插入合适位置
			data[low] = tmp;
			System.out.println(java.util.Arrays.toString(data));
		}
	}
	public static void main(String[] args)
	{
		DataWrap[] data = {
			new DataWrap(9 , ""),
			new DataWrap(-16 , ""),
			new DataWrap(21 , "*"),
			new DataWrap(23 , ""),
			new DataWrap(-30 , ""),
			new DataWrap(-49 , ""),
			new DataWrap(21 , ""),
			new DataWrap(30 , "*"),
			new DataWrap(30 , "")
		};
		System.out.println("排序之前:\n"
			+ java.util.Arrays.toString(data));
		binaryInsertSort(data);
		System.out.println("排序之后:\n" 
			+ java.util.Arrays.toString(data));
	}
}

三、shell排序
希尔排序基本思想
    先取一个小于n的整数d1作为第一个增量,把文件的全部记录分成d1个组。所有距离为dl的倍数的记录放在同一个组中。先在各组内进行直接插人排序;然后,取第二个增量d2<d1重复上述的分组和排序,直至所取的增量dt=1(dt<dt-l<…<d2<d1),即所有记录放在同一组中进行直接插入排序为止。
  该方法实质上是一种分组插入方法。
算法步骤
  Step1 将n个元素个数列分为5个小组,在每个小组内按直接插入法排序;
  step2 在第i步,分组个数取 di+1 =(di +1)/2 {9,5,3,2,1};相临两组之间的对应元素进行比较,如果ai>aj,则交换它们的位置;
  Step3 当dK = 1的循环过程完成后,排序过程结束。
Shell排序的时间性能优于直接插入排序
  希尔排序的时间性能优于直接插入排序的原因:
  ①当文件初态基本有序时直接插入排序所需的比较和移动次数均较少。
  ②当n值较小时,n和n的差别也较小,即直接插入排序的最好时间复杂度O(n)和最坏时间复杂度0(n)差别不大。
  ③在希尔排序开始时增量较大,分组较多,每组的记录数目少,故各组内直接插入较快,后来增量di逐渐缩小,分组数逐渐减少,而各组的记录数目逐渐增多,但由于已经按di-1作为距离排过序,使文件较接近于有序状态,所以新的一趟排序过程也较快。
  因此,希尔排序在效率上较直接插人排序有较大的改进。
稳定性
  希尔排序是不稳定的。


public class ShellSort
{
	public static void shellSort(DataWrap[] data) 
	{
		System.out.println("开始排序:");
		int arrayLength = data.length;
		//h变量保存可变增量
		int h = 1;
		//按h * 3 + 1得到增量序列的最大值
		while(h <= arrayLength / 3)
		{
			h = h * 3 + 1;
		}
		while(h > 0)
		{
			System.out.println("===h的值:" + h + "===");
			for (int i = h ; i < arrayLength ; i++ )
			{
				//当整体后移时,保证data[i]的值不会丢失
				DataWrap tmp = data[i];
				//i索引处的值已经比前面所有值都大,表明已经有序,无需插入
				//(i-1索引之前的数据已经有序的,i-1索引处元素的值就是最大值)
				if (data[i].compareTo(data[i - h]) < 0)
				{
					int j = i - h;
					//整体后移h格
					for ( ; j >= 0 && data[j].compareTo(tmp) > 0 ; j-=h)
					{
						data[j + h] = data[j];
					}
					//最后将tmp的值插入合适位置
					data[j + h] = tmp;
				}
				System.out.println(java.util.Arrays.toString(data));
			}
			h = (h - 1) / 3;
		}
	}
	public static void main(String[] args)
	{
		DataWrap[] data = {
			new DataWrap(9 , ""),
			new DataWrap(-16 , ""),
			new DataWrap(21 , "*"),
			new DataWrap(23 , ""),
			new DataWrap(-30 , ""),
			new DataWrap(-49 , ""),
			new DataWrap(21 , ""),
			new DataWrap(30 , "*"),
			new DataWrap(30 , ""),
		};
		System.out.println("排序之前:\n"
			+ java.util.Arrays.toString(data));
		shellSort(data);
		System.out.println("排序之后:\n" 
			+ java.util.Arrays.toString(data));
	}
}
分享到:
评论

相关推荐

    java实现常见排序算法

    Java中常见的插入排序实现有三种:直接插入排序、折半插入排序和希尔排序。 1. **直接插入排序**: 直接插入排序的基本思想是将一个记录插入到已经排好序的有序表中,从而得到一个新的、记录数增1的有序表。在Java...

    几种经典排序算法的Java实现

    ### 几种经典排序算法的Java实现 ...通过上述介绍和示例代码,我们可以看出直接插入排序和折半插入排序的具体实现方式及其性能特点。在实际应用中,应根据具体需求选择合适的排序算法以达到最优效果。

    各种排序算法Demo

    折半插入排序是在直接插入排序的基础上优化,利用二分查找的方法在已排序的序列中寻找插入位置,从而减少比较次数,提高了效率。 3. 希尔排序(Shell排序): 希尔排序是插入排序的一种更高效的改进版本,由...

    java实现数据结构内部排序.pdf

    这个PDF文档可能涵盖了多种内部排序算法的Java实现,包括直接插入排序、折半插入排序、希尔排序以及快速排序。以下是对这些排序算法的详细说明: 1. **直接插入排序(InsertSort)**: 直接插入排序是一种简单的...

    java常见的排序算法源代码

    【Java排序算法详解】 在Java编程中,排序算法是数据结构和算法领域的重要组成部分,用于组织和优化数据。本文将详细介绍几种常见的Java排序算法,并提供相应的源代码。 1. **直接插入排序(Insertion Sort)** ...

    所有排序的写法(Java).zip

    这里我们主要探讨的是使用Java实现的几种经典排序算法,包括直接选择排序、堆排序、冒泡排序、快速排序、直接插入排序、折半插入排序、Shell排序、归并排序、桶式排序和基数排序。这些算法各有特点,适用于不同的...

    Java 排序大全排序大全

    根据给定的信息,本文将详细解释Java中几种重要的排序算法:直接插入排序、折半插入排序、Shell排序、冒泡排序、快速排序、选择排序以及堆排序。 ### 直接插入排序 直接插入排序的基本思想是:将一个记录插入到...

    常用排序算法java版

    选择排序(直接选择排序,堆排序) 交换排序(冒泡排序,快速排序) 插入排序(直接插入排序,折半插入排序,Shell排序) 归并排序 桶式排序 基数排序

    java基础数据结构-排序算法

    - **Shell排序**:通过先将整个待排序的记录分割成为若干子序列分别进行直接插入排序,最后再对整个序列进行一次直接插入排序。时间复杂度取决于增量序列的选择。 - **归并排序**:也是一种分治算法,将数组分成两...

    数据结构排序查找常用算法实现.zip

    堆排序的时间复杂度为O(n log n),空间复杂度为O(1),是原地排序算法。 7. 折半查找(Binary Search): 折半查找,也称为二分查找,适用于已排序的列表。它通过不断缩小查找范围,每次将查找区间减半,直到找到...

    所有排序方法.zip

    5. **折半插入排序**(BinaryInsertSort.java):在直接插入排序的基础上,通过二分查找找到插入位置,减少了比较次数,提高了效率。但基本时间复杂度仍为O(n^2)。 6. **Shell排序**(ShellSort.java):Shell排序...

    java几种排序算法的实现及简单分析

    折半插入排序是在直接插入排序的基础上,增加了二分查找的优化,提高了插入元素时的效率。在`binInsertSort`方法中,使用二分查找确定元素的插入位置,从而减少平均时间复杂度。 3. **希尔排序(Shell Sort)**: ...

    详细总结各种排序算法(Java实现)

    冒泡排序是最简单的排序算法之一,通过不断交换相邻的逆序元素来实现排序。它的时间复杂度为O(n²),空间复杂度为O(1),是稳定的排序。 ```java public class BubbleSort { public void sort(int[] a) { int temp...

    915考试大纲1

    常见的内部排序算法有直接插入排序、冒泡排序、简单选择排序、Shell排序、快速排序、堆排序、归并排序和基数排序。排序算法的稳定性是指相等元素的相对顺序是否在排序后保持不变。每种排序算法都有其特点和适用场景...

    数据结构考研复习

    - **折半插入排序:** 类似于直接插入排序,但在插入前使用折半查找确定插入位置。 - **气泡排序(bubblesort):** 通过重复地走访过要排序的数列,一次比较两个元素,如果他们的顺序错误就把他们交换过来。 - **...

Global site tag (gtag.js) - Google Analytics