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

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

大家好,又见面了,我是全栈君,今天给大家准备了Idea注册码。

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个索引上的元素要插入的位置是:1918 562 442 531 210 216 931 706 333 132 第2个索引上的元素要插入的位置是:2918 562 442 531 210 216 931 706 333 132 第3个索引上的元素要插入的位置是:2918 562 531 442 210 216 931 706 333 132 第4个索引上的元素要插入的位置是:4918 562 531 442 210 216 931 706 333 132 第5个索引上的元素要插入的位置是:4918 562 531 442 216 210 931 706 333 132 第6个索引上的元素要插入的位置是:0931 918 562 531 442 216 210 706 333 132 第7个索引上的元素要插入的位置是:2931 918 706 562 531 442 216 210 333 132 第8个索引上的元素要插入的位置是:6931 918 706 562 531 442 333 216 210 132 第9个索引上的元素要插入的位置是:9 设值次数(setValueCount)=====> 24931 918 706 562 531 442 333 216 210 132 

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

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/117937.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • 什么叫应用程序域?(zhuan)

    什么叫应用程序域?(zhuan) 一:应用程序域介绍:    "域",就是范围,环境,边界的意思,那么什么是应用程序域,官方给出的是这样的解释:操作系

  • 伽马校正_液晶电视伽马校正

    伽马校正_液晶电视伽马校正注:本文为博主参考书籍和他人文章并加上自己的理解所编,作为学习笔记使用并将其分享出去供大家学习。若涉及到引用您的文章内容请评论区告知!如有错误欢迎指正!一、伽马校正简介    射伽马校正是图像预处理阶段经常使用的一个非线性算子,它可以去除输入辐射量和量化的像素值之间的非线性映。所谓伽玛校正就是对图像的伽玛曲线进行编辑,以对图像进行非线性色调编辑的方法,检出图像信号中的深色部分和浅色…

  • Qt 用QMediaPlayer实现简易播放器

    Qt 用QMediaPlayer实现简易播放器QtMultimedia是Qt的一个重要模块,它提供了许多c++类和QML模块来进行多媒体内容的展示和处理,还提供了一些访问录音机和摄像头的必要的api。本篇中,主要讲述的是C++的实现,而不是QM

  • 强化学习 — MCTS

    强化学习 — MCTS目录Simulation-BasedSearchMCSearchMCTS上线置信区间算法UCT棋类游戏MCTS搜索Simulation-BasedSearch基于仿真的搜索包含两点:一个是simulation,其次是search。simulation是基于强化学习model进行采样,得到样本数据。但这不是基于和环境交互获得的真实数据。search则是为了利用样本结果来帮我们计算应该采用什么动作,以实现长期利益最大化要理解什么是Simulation-BasedSearch,首先要明白什么是for

  • 电路的kvl方程_电路分析依据两类约束

    电路的kvl方程_电路分析依据两类约束1.2b方程2.举例举例2

  • 工厂模式和抽象工厂模式的区别_工厂模式代码

    工厂模式和抽象工厂模式的区别_工厂模式代码工厂模式 Factory Method动机模式定义实例结构图要点总结笔记动机在软件系统中,经常面临着创建对象的工作,由于需求的变换,需要创建的对象的具体类型经常变换。如何应对这种变换?如何绕过常规的对象创建方法(new),提供一种”封装机制“来避免客户程序和这种”具体对象创建工作“的紧耦合模式定义定义一个用于创建对象的接口,让子类决定实例化哪一个类。Factory Method使得一个类的实例化延迟(目的:解耦,手段:虚函数)到子类实例朴素class ISplitter{public:

发表回复

您的电子邮箱地址不会被公开。

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