Java 实现二分(折半)插入排序

设有一个序列a[0],a[1]...a[n];其中a[i-1]前是已经有序的,当插入时a[i]时,利用二分法搜索a[i]插入的位置

效率:O(N^2),对于初始基本有序的序列,效率上不如直接插入排序;对于随机无序的序列,效率比直接插入排序要高

/*
 * 二分(折半)插入排序
 * 设有一个序列a[0],a[1]...a[n];其中a[i-1]前是已经有序的,当插入时a[i]时,利用二分法搜索a[i]插入的位置
 */
public class BinaryInsertSort {

	public static void main(String[] args) {
		int len = 10;
		int[] ary = new int[len];
		Random random = new Random();
		for (int j = 0; j < len; j++) {
			ary[j] = random.nextInt(1000);
		}
		binaryInsert(ary);
		/*
		 * 复杂度分析: 最佳情况,即都已经排好序,则无需右移,此时时间复杂度为:O(n lg n) 最差情况,全部逆序,此时复杂度为O(n^2)
		 *  无法将最差情况的复杂度提升到O(n|logn)。
		 */
		// 打印数组
		printArray(ary);
	}
	/**
	 * 插入排序
	 * @param ary
	 */
	private static void binaryInsert(int[] ary) {
		int setValueCount = 0;
		// 从数组第二个元素开始排序,因为第一个元素本身肯定是已经排好序的
		for (int j = 1; j < ary.length; j++) {// 复杂度 n
			// 保存当前值
			int key = ary[j];
			// ? 利用二分查找定位插入位置
//			int index = binarySearchAsc(ary, ary[j], 0, j - 1);// 复杂度:O(logn)
//			int index = binarySearchDesc(ary, ary[j], 0, j - 1);// 复杂度:O(logn)
			int index = binarySearchDesc2(ary, ary[j], 0, j - 1);// 复杂度:O(logn)
			printArray(ary);
			System.out.println("第" + j +"个索引上的元素要插入的位置是:" + index);
			// 将目标插入位置,同时右移目标位置右边的元素
			for (int i = j; i > index; i--) {// 复杂度,最差情况:(n-1)+(n-2)+...+n/2=O(n^2)
				ary[i] = ary[i - 1]; //i-1 <==> index
				setValueCount++;
			}
			ary[index] = key;
			setValueCount++;
		}
		System.out.println("\n 设值次数(setValueCount)=====> " + setValueCount);
	}

	/**
	 * 二分查找 升序 递归
	 *
	 * @param ary
	 *            给定已排序的待查数组
	 * @param target
	 *            查找目标
	 * @param from
	 *            当前查找的范围起点
	 * @param to
	 *            当前查找的返回终点
	 * @return 返回目标在数组中,按顺序应在的位置
	 */
	private static int binarySearchAsc(int[] ary, int target, int from, int to) {
		int range = to - from;
		// 如果范围大于0,即存在两个以上的元素,则继续拆分
		if (range > 0) {
			// 选定中间位
			int mid = (to + from) / 2;
			// 如果临界位不满足,则继续二分查找
			if (ary[mid] > target) {
				/*
				 * mid > target, 升序规则,target较小,应交换位置 前置, 即target定位在mid位置上,
				 * 根据 查找思想, 从from到 mid-1认为有序, 所以to=mid-1
				 */
				return binarySearchAsc(ary, target, from, mid - 1);
			} else {
				/*
				 * mid < target, 升序规则,target较大,不交换位置,查找比较的起始位置应为mid+1
				 */
				return binarySearchAsc(ary, target, mid + 1, to);
			}
		} else {
			if (ary[from] > target) {//如 5,4, 要插入的是4
				return from;
			} else {
				return from + 1;
			}
		}
	}
	/**
	 * 二分查找 降序, 递归
	 */
	private static int binarySearchDesc(int[] ary, int target, int from, int to) {
		int range = to - from;
		if (range > 0) {
			int mid = (from + to) >>> 1;
			if (ary[mid] > target) {
				return binarySearchDesc(ary, target, mid + 1, to);
			} else {
				return binarySearchDesc(ary, target, from, mid - 1);
			}
		} else {
			if (ary[from] > target) {//如 5,4, 要插入的是4
				return from + 1;
			} else {
				return from;
			}
		}
	}

	/**
	 * 二分查找 降序, 非递归
	 */
	private static int binarySearchDesc2(int[] ary, int target, int from, int to) {
//		while(from < to) {
		for (; from < to; ) {
			int mid = (from + to) >>> 1;
			if (ary[mid] > target) {
				from = mid + 1;
			} else {
				to  = mid -1;
			}
		}
		//from <==> to;
		if (ary[from] > target) {//如 5,4, 要插入的是4
			return from + 1;
		} else {
			return from;
		}
	}

	private static void printArray(int[] ary) {
		for (int i : ary) {
			System.out.print(i + " ");
		}
	}

}

打印

918 562 442 531 210 216 931 706 333 132 第1个索引上的元素要插入的位置是:1
918 562 442 531 210 216 931 706 333 132 第2个索引上的元素要插入的位置是:2
918 562 442 531 210 216 931 706 333 132 第3个索引上的元素要插入的位置是:2
918 562 531 442 210 216 931 706 333 132 第4个索引上的元素要插入的位置是:4
918 562 531 442 210 216 931 706 333 132 第5个索引上的元素要插入的位置是:4
918 562 531 442 216 210 931 706 333 132 第6个索引上的元素要插入的位置是:0
931 918 562 531 442 216 210 706 333 132 第7个索引上的元素要插入的位置是:2
931 918 706 562 531 442 216 210 333 132 第8个索引上的元素要插入的位置是:6
931 918 706 562 531 442 333 216 210 132 第9个索引上的元素要插入的位置是:9

 设值次数(setValueCount)=====> 24
931 918 706 562 531 442 333 216 210 132 
时间: 2024-11-09 01:41:15

Java 实现二分(折半)插入排序的相关文章

排序系列 之 折半插入排序算法 —— Java实现

基本思想: 折半插入算法是对直接插入排序算法的改进,排序原理同直接插入算法: 把n个待排序的元素看成一个有序表和一个无序表,开始时有序表中只有一个元素,无序表中有n-1个元素:排序过程即每次从无序表中取出第一个元素,将它插入到有序表中,使之成为新的有序表,重复n-1次完成整个排序过程. 与直接插入算法的区别在于:在有序表中寻找待排序数据的正确位置时,使用了折半查找/二分查找. 实例: (参考直接插入排序算法:http://www.cnblogs.com/snowcan/p/6244128.htm

直接插入排序与折半插入排序——Java实现

1.直接插入排序 1)原理说明:直接插入排序是一种最简单的插入排序方法,它的基本思想是,仅有一个元素的序列总是有序的,因此,对n 个记录的序列,可从第二个元素开始直到第n 个元素,逐个向有序序列中执行插入操作,从而得到n 个元素按关键字有序的序列. 2)代码实现如下: package com.test.sort; public class InsertSort { private static void sort(int[] data) { for (int i = 1; i < data.le

数据结构之排序算法Java实现(6)—— 插入类排序之折半插入排序算法

折半插入排序是对直接插入排序进行了改进,在寻找插入点时使用二分查找算法,提高了查询效率. 升序排序: /** * 折半插入排序 * 升序排序 */ @Override public <T extends Comparable<? super T>> void sortByAsc(T[] data) { for(int i = 1;i < data.length;i++ ){ if(data[i].compareTo(data[i - 1]) < 0){ /**记录i的值

Java使用二分插入排序竟然和直接插入排序速度相差不多

?? Java使用二分插入排序竟然和直接插入排序速度相差不多 之前测试过Python使用二分插入排序竟然比直接插入排序快99倍! 现在测试下 Java,Linux测试结果如下: javac test.java java testInsertSort total milliseconds:15769InsertSortWithBinarySerach total milliseconds:15657 程序如下: import java.util.Date; public class test{ p

从二分查找到折半插入排序

目录 从二分查找到折半插入排序 折半插入排序性能 Code 从二分查找到折半插入排序 回忆直接插入排序的过程,发现每趟排序中进行了两个动作: \1. 从左边的已排序序列中找寻插入位置. \2. 给插入位置腾出空间,将插入元素复制到表中的插入位置. 步骤一在直接插入排序中是一个"Linear Search"顺序查找过程,而我们知道二分查找比顺序查找更优秀. 折半插入排序性能 Space Complexity: S(n)=O(1) Time Complexity: T(n)=O(\(n^2

算法学习之排序算法:插入排序(直接插入排序、折半插入排序、2-路插入排序)

引言: 插入排序作为最简单易于理解的排序算法,基本实现比较简单.本文详细介绍直接插入排序,并给出实现,简单的介绍折半插入排序,并给出2-路插入排序和表插入排序两种插入排序,但并未给出具体实现. 一.直接插入排序 直接插入排序的基本操作是将一个记录插入到已排好序的有序表中,从而得到一个新的.记录数增1的有序表. 算法描述: 步骤1.将待排序的一组记录中的第1个记录拿出来作为一组有序的记录(当然此时该组记录仅有1个记录). 步骤2.依次将待排序的一组记录中的记录拿出来插入到前面已排好序的记录中. 步

排序算法总结之折半插入排序

基本思想 折半插入排序是对直接插入排序的简单改进,对于直接插入排序而言,当第i-1趟需要将第i个元素插入前面的0~i-1个元素序列中时,总是需要从i-1个元素开始,逐个比较每个元素,直到找到它的位置.这显然没有利用前面0~i-1个元素已经有序这个特点,而折半插入排序则改进了这一点. 对于折半插入排序而言,当需要插入第i个元素时,它不会逐个进行比较每个元素,而是: (1)计算0~i-1索引的中间点,也就是用i索引处的元素和(0+i-1)/2索引处的元素进行比较,如果i索引处的元素值大,就直接在(0

排序算法2--插入排序--折半插入排序

折半插入排序 折半插入排序(binary insertion sort)是对插入排序算法的一种改进,所谓排序算法过程,就是不断的依次将元素插入前面已排好序的序列中. 在将一个新元素插入已排好序的数组的过程中,寻找插入点时,将待插入区域的首元素设置为a[low],末元素设置为a[high],则轮比较时将待插入元素与a[m],其中m=(low+high)/2相比较,如果比参考元素小,则选择a[low]到a[m-1]为新的插入区域(即high=m-1),否则选择a[m+1]到a[high]为新的插入区

插入排序(直接插入排序、折半插入排序)

一.直接插入排序 package algorithm.sort.compare.insert; import java.util.Arrays; public class DirectInsertSort { public static void main(String[] args) { int[] arrayA = new int[] {11, 213, 134, 65, 77, 78, 23, 43}; directInsertSort (arrayA); System.out.print