Java使用二分法进行查找和排序的示例
程序员文章站
2024-03-09 13:13:47
实现二分法查找
二分法查找,需要数组内是一个有序的序列
二分查找比线性查找:数组的元素数越多,效率提高的越明显
二分查找的效率表示:o(log2n) n在2的m次幂范...
实现二分法查找
二分法查找,需要数组内是一个有序的序列
二分查找比线性查找:数组的元素数越多,效率提高的越明显
二分查找的效率表示:o(log2n) n在2的m次幂范围,那查找的次数最大就是m, log2n表示2的m次幂等于n, 省略常数,简写成o(logn)
如有一个200个元素的有序数组,那么二分查找的最大次数:
2^7=128, 2^8=256, 可以看出7次幂达不到200,8次幂包括, 所以最大查找次数就等于8
//循环,二分查找 static int binarysearch(int[] array, int data) { int start = 0; int end = array.length - 1; int mid = -1; while (start <= end) { system.out.println("查找次数"); mid = (start + end) >>> 1; if (array[mid] < data) { start = mid + 1; } else if (array[mid] > data) { end = mid - 1; } else { return mid; } system.out.println("start=" + start+",end="+end+",mid="+mid); } return -1; }
//递归二分查找 初始start=0, end = array.length - 1 static int binarysearch4recursion(int[] array, int data, int start, int end) { int mid = -1; system.out.println("查找次数"); if (start > end) { return mid; } mid = (start + end) >>> 1; if (array[mid] < data) { return binarysearch4recursion(array, data, mid + 1, end); } else if (array[mid] > data) { return binarysearch4recursion(array, data, start, mid - 1); } else { return mid; } }
二分法插入排序
设有一个序列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
推荐阅读
-
Java使用二分法进行查找和排序的示例
-
使用Java的Lucene搜索工具对检索结果进行分组和分页
-
Java的微信开发中使用XML格式和JSON格式数据的示例
-
使用CXF和Jersey框架来进行Java的WebService编程
-
Java基础知识回顾第一篇 - 数组和List之间的相互转换 | 二分法查找 | 冒泡排序 博客分类: Java基础知识回顾 冒泡排序二分法查找Java基础
-
使用CXF和Jersey框架来进行Java的WebService编程
-
使用Java代码进行因数分解和求最小公倍数的示例
-
Java中使用DOM和SAX解析XML文件的方法示例
-
使用Java对数据库进行基本的查询和更新操作
-
使用Java代码进行因数分解和求最小公倍数的示例