算法导论Java实现-二分查找运用(习题2.3-7)

简介:

 
 
  1. package lhz.algorithm.chapter.two; 
  2.  
  3. /** 
  4.  * 《算法导论》,习题2.3-7  
  5.  * Describe a Θ(n lg n)-time algorithm that, given a set S of n 
  6.  * integers and another integer x, determines whether or not there exist two 
  7.  * elements in S whose sum is exactly x. 
  8.  * @author lihzh(苦逼coder) 
  9. * 本文地址:http://mushiqianmeng.blog.51cto.com/3970029/732739  
  10. */ 
  11. public class Excercise237 { 
  12.  
  13.     private static int[] input = new int[] { 21549867103 }; 
  14.     private static int sum = 11
  15.  
  16.     public static void main(String[] args) { 
  17.         // 先利用合并排序,将给定数组排好序 
  18.         mergeSort(input);// 复杂度:Θ(nlgn) 
  19.         boolean isExist = false
  20.         // 设置循环结束位,每次找到符合条件的元素后,改变循环结束条件 
  21.         // 否则会出现11 = 1 + 10,11 = 10 + 1的重复情况。 
  22.         int end = input.length; 
  23.         for (int i = 0; i < end; i++) {// 复杂度:Θ(n) 
  24.             int temp = sum - input[i]; 
  25.             // 利用二分查找,查询temp是否在数组中。 
  26.             Integer index = binarySearch(input, temp, 0, input.length - 1);// 复杂度:Θ(lgn) 
  27.             if (index != null && index != i) {// 不等于本身 
  28.                 System.out.println(sum + " = " + input[i] + " + " + temp); 
  29.                 end = index; 
  30.                 isExist = true
  31.             } 
  32.         } 
  33.         if (!isExist) { 
  34.             System.out.println("数组不存在两个数的和为:" + sum); 
  35.         } 
  36.         /* 
  37.          * 复杂度分析: 由注释可见,复杂度为:Θ(nlgn) 
  38.          */ 
  39.     } 
  40.  
  41.     /** 
  42.      * 合并排序,复杂度:Θ(nlgn) 
  43.      *  
  44.      * @param array 
  45.      * @return 
  46.      */ 
  47.     private static int[] mergeSort(int[] array) { 
  48.         // 如果数组的长度大于1,继续分解数组 
  49.         if (array.length > 1) { 
  50.             int leftLength = array.length / 2
  51.             int rightLength = array.length - leftLength; 
  52.             // 创建两个新的数组 
  53.             int[] left = new int[leftLength]; 
  54.             int[] right = new int[rightLength]; 
  55.             // 将array中的值分别对应复制到两个子数组中 
  56.             for (int i = 0; i < leftLength; i++) { 
  57.                 left[i] = array[i]; 
  58.             } 
  59.             for (int i = 0; i < rightLength; i++) { 
  60.                 right[i] = array[leftLength + i]; 
  61.             } 
  62.             // 递归利用合并排序,排序子数组 
  63.             left = mergeSort(left); 
  64.             right = mergeSort(right); 
  65.             // 设置初始索引 
  66.             int i = 0
  67.             int j = 0
  68.             for (int k = 0; k < array.length; k++) { 
  69.                 // 如果左边数据索引到达边界则取右边的值 
  70.                 if (i == leftLength && j < rightLength) { 
  71.                     array[k] = right[j]; 
  72.                     j++; 
  73.                     // 如果右边数组索引到达边界,取左数组的值 
  74.                 } else if (i < leftLength && j == rightLength) { 
  75.                     array[k] = left[i]; 
  76.                     i++; 
  77.                     // 如果均为到达则取,较小的值 
  78.                 } else if (i < leftLength && j < rightLength) { 
  79.                     if (left[i] > right[j]) { 
  80.                         array[k] = right[j]; 
  81.                         j++; 
  82.                     } else { 
  83.                         array[k] = left[i]; 
  84.                         i++; 
  85.                     } 
  86.                 } 
  87.             } 
  88.         } 
  89.         return array; 
  90.     } 
  91.  
  92.     /** 
  93.      * 二分查找,复杂度Θ(lg n) 
  94.      *  
  95.      * @param input 
  96.      * @param target 
  97.      * @param from 
  98.      * @param to 
  99.      * @return 
  100.      */ 
  101.     private static Integer binarySearch(int[] input, int target, int from, 
  102.             int to) { 
  103.         int range = to - from; 
  104.         // 如果范围大于0,即存在两个以上的元素,则继续拆分 
  105.         if (range > 0) { 
  106.             // 选定中间位 
  107.             int mid = (to + from) / 2
  108.             // 先判断两个二分的临界位 
  109.             if (input[mid] == target) { 
  110.                 return mid; 
  111.             } 
  112.             // 如果临界位不满足,则继续二分查找 
  113.             if (input[mid] > target) { 
  114.                 return binarySearch(input, target, from, mid - 1); 
  115.             } else { 
  116.                 return binarySearch(input, target, mid + 1, to); 
  117.             } 
  118.         } else { 
  119.             if (input[from] == target) { 
  120.                 return from; 
  121.             } else { 
  122.                 return null
  123.             } 
  124.         } 
  125.     } 

 


     本文转自mushiqianmeng 51CTO博客,原文链接:http://blog.51cto.com/mushiqianmeng/732739,如需转载请自行联系原作者




相关文章
|
6天前
|
存储 算法 安全
探究‘公司禁用 U 盘’背后的哈希表算法与 Java 实现
在数字化办公时代,信息安全至关重要。许多公司采取“禁用U盘”策略,利用哈希表算法高效管理外接设备的接入权限。哈希表通过哈希函数将设备标识映射到数组索引,快速判断U盘是否授权。例如,公司预先将允许的U盘标识存入哈希表,新设备接入时迅速验证,未授权则禁止传输并报警。这有效防止恶意软件和数据泄露,保障企业信息安全。 代码示例展示了如何用Java实现简单的哈希表,模拟公司U盘管控场景。哈希表不仅用于设备管理,还在文件索引、用户权限等多方面助力信息安全防线的构建,为企业数字化进程保驾护航。
|
3月前
|
存储 人工智能 算法
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
这篇文章详细介绍了Dijkstra和Floyd算法,这两种算法分别用于解决单源和多源最短路径问题,并且提供了Java语言的实现代码。
109 3
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
|
5天前
|
存储 算法 测试技术
【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
本任务旨在实现二叉树的遍历,包括先序、中序、后序和层次遍历。首先介绍了二叉树的基本概念与结构定义,并通过C++代码示例展示了如何定义二叉树节点及构建二叉树。接着详细讲解了四种遍历方法的递归实现逻辑,以及层次遍历中队列的应用。最后提供了测试用例和预期输出,确保代码正确性。通过这些内容,帮助读者理解并掌握二叉树遍历的核心思想与实现技巧。
24 2
|
19天前
|
算法 索引
【算法】——二分查找合集
二分查找基础模版和进阶模版,查找元素位置,搜索插入位置,x的平方根,山脉数组的峰顶索引,寻找峰值,点名
|
5月前
|
算法
【算法】二分查找——在排序数组中查找元素的第一个和最后一个位置
【算法】二分查找——在排序数组中查找元素的第一个和最后一个位置
|
2月前
|
算法 安全 NoSQL
2024重生之回溯数据结构与算法系列学习之顺序表习题精讲【无论是王道考研人还真爱粉都能包会的;不然别给我家鸽鸽丢脸好嘛?】
顺序表的定义和基本操作之插入;删除;按值查找;按位查找习题精讲等具体详解步骤以及举例说明
|
3月前
|
Java
在 Java 中实现二分查找法
【10月更文挑战第9天】
42 1
|
3月前
|
算法 C# 索引
C#二分查找算法
C#二分查找算法
|
3月前
|
算法 Java
java冒泡排序与二分查找(详解)
java冒泡排序与二分查找(详解)
52 4
|
3月前
|
Java 数据安全/隐私保护
JAVA经典习题详解
JAVA经典习题详解
30 4