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

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

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

全栈程序员社区此处内容已经被作者隐藏,请输入验证码查看内容
验证码:
请关注本站微信公众号,回复“验证码”,获取验证码。在微信里搜索“全栈程序员社区”或者“www_javaforall_cn”或者微信扫描右侧二维码都可以关注本站微信公众号。

设有一个序列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 

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

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

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Linux 修改文件权限

    Linux 修改文件权限目录:1、介绍:2、权限说明3、用户4、权限设置5、权限设置格式:(1)增加权限:(2)撤销权限(3)无任何权限1、介绍:​ Linux系统中,每个文件或目录都有访问许可权限,用它来确定以何种方式对文件或目录进行访问和操作。在Linux中,如果要对文件的权限进行修改,那么可在终端中使用chmod命令对其文件的权限进行修改,但是chmod命令修改文件权限有两种方式:1、字母法,2、数字法2、权限说明​ (1)只读:表示只允许读取内容,而禁止其对该文件做其他任何操作​ 字母法:‘r’

    2025年10月29日
    3
  • [Python人工智能] 九.gensim词向量Word2Vec安装及《庆余年》中文短文本相似度计算

    [Python人工智能] 九.gensim词向量Word2Vec安装及《庆余年》中文短文本相似度计算从本专栏开始,作者正式开始研究Python深度学习、神经网络及人工智能相关知识。前一篇详细讲解了卷积神经网络CNN原理,并通过TensorFlow编写CNN实现了MNIST分类学习案例。本篇文章将分享gensim词向量Word2Vec安装、基础用法,并实现《庆余年》中文短文本相似度计算及多个案例。本专栏主要结合作者之前的博客、AI经验和相关文章及论文介绍,后面随着深入会讲解更多的Python人工智能案例及应用。基础性文章,希望对您有所帮助~

    2022年6月8日
    31
  • JAX-WS – Soap详解[通俗易懂]

    JAX-WS – Soap详解[通俗易懂]一、创建Soap //创建消息工厂 MessageFactoryfactory=MessageFactory.newInstance(); //通过工厂创建Soap的消息 SOAPMessagemessage=factory.createMessage(); //获取SoapPart SOAPPartpart=message.getSO

    2022年7月15日
    19
  • java webservice接口开发教程_JAVA入门教程

    java webservice接口开发教程_JAVA入门教程写在前面的话:当两个人碰面后,产生了好感,如果需要得到双方的信息,那么双方的交流是必不可少的!应用程序也如此,各个应用程序之间的交流就需要WebService来作为相互交流的桥梁!项目目的:程序A调用程序B中的方法C…首先申明:本次需要采用的JDK版本为jdk1.6+,也就是要java6+才能看到效果,java5没试过…我认为,升级一下JDK版本比起安装什么Axis以及导入一堆jar包和配…

    2025年5月25日
    3
  • java中遍历数组_java循环取数组值

    java中遍历数组_java循环取数组值第一种方法(传统方法)inta[]={1,2,3,4,5};for(inti=0;i<a.length;i++){System.out.println(a[i]);}第二种方法(非传统方法)inta[]={1,2,3,4,5};for(inti:a){System.out.println(i);}注意:其中i的数据类型要和数组a的数组类型保持一致。第三种方…

    2022年9月19日
    5
  • laravel-admin 安装(总结)

    laravel-admin 安装(总结)

    2021年10月29日
    42

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

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