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)


相关推荐

  • GridView编辑删除操作

    GridView编辑删除操作

    2021年11月28日
  • mysql解锁命令_mysql锁表查询和解锁操作

    mysql解锁命令_mysql锁表查询和解锁操作解除正在死锁的状态有两种方法:第一种:1.查询是否锁表showOPENTABLESwhereIn_use>0;2.查询进程(如果您有SUPER权限,您可以看到所有线程。否则,您只能看到您自己的线程)showprocesslist3.杀死进程id(就是上面命令的id列)killid第二种:1.查看下在锁的事务SELECT*FROMINFORMATION_SCHEMA.IN…

  • 电商新宠—广告电商,转化产品的流量聚体地「建议收藏」

    电商新宠—广告电商,转化产品的流量聚体地「建议收藏」现在市面上通过看广告赚点零花钱的项目也是片地一把抓,在各大平台看广告,间接给平台赚钱,却不能给自己带来一些好处;而真正的并没有让消费者能长期的去坚持去做,一个月下来看广告收益也就十多二十元块钱,使大多数平台变得暗淡下去,最后无人问津。目前又听说在市面上流传了一个很火热的广告变现模式——广告电商,结合了“社交电商+广告分佣”,通过在平台购买商品,赠送同等或者一定量的积分,达到不同的门槛,可以根据不同的积分门槛看不同的广告(每天3分钟),实现广告变现,提现到微信、支付宝和对接的第三方支付服务平台。最终实现广告主

  • mac 显示 sh-3.2# 的

    mac 显示 sh-3.2# 的

  • 基于javaEE的医院病历管理系统的设计与实现[通俗易懂]

    网络的高速发展,促使着数字化医院的建设,现如今大多数医院已经在使用病历管理系统来管理患者电子病历。在医院中,病历记录了医生和患者的诊疗过程,医生可以通过之前病历记载,快速诊断患者,所以病历是医院的重要资产。使用计算机可以提高病历质量,方便存储、查阅、检索等,从而提高病案管理效率,实现病历信息同时异地共享和反复利用。电子病历的推广应用已经势不可挡,未来电子病历需求更高,应用也将继续成熟,市场的竞争也更加激烈。本次毕业设计的题目是基于javaEE的医院病历管理系统的设计与实现。本系统主要运用java编程语言、基

  • 常见端口渗透总结[通俗易懂]

    常见端口渗透总结[通俗易懂]文章目录0x00背景服务默认端口爆破0x01实战测试文件共享服务端口渗透ftp服务NFS服务Samba服务LDAP协议远程连接服务端口渗透SSH服务Telnet服务Windows远程连接VNC服务Pcanywhere服务Web应用服务端口渗透IIS服务Apache/Tomcat/Nginx/Axis2WebLogicJbossWebsphereGlassFishJenkinsResinJettyLotus数据库服务端口渗透MySQL数据库MSSQL数据库Oracle数据库PostgreSQL数据库Mon

发表回复

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

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