二分查找也称折半查找(Binary Search),它是一种效率较高的查找方法。但是,折半查找要求线性表必须采用顺序存储结构,而且表中元素按关键字有序排列。
二分查找优缺点
优点是比较次数少,查找速度快,平均性能好;
其缺点是要求待查表为有序表,且插入删除困难。
因此,折半查找方法适用于不经常变动而查找频繁的有序列表。
使用条件:查找序列是顺序结构,有序。
过程
首先,假设表中元素是按升序排列,将表中间位置记录的关键字与查找关键字比较,如果两者相等,则查找成功;否则利用中间位置记录将表分成前、后两个子表,如果中间位置记录的关键字大于查找关键字,则进一步查找前一子表,否则进一步查找后一子表。重复以上过程,直到找到满足条件的记录,使查找成功,或直到子表不存在为止,此时查找不成功。
利用循环的方式实现二分法查找
public class BinarySearch {
public static void main(String[] args) {
// 生成一个随机数组 int[] array = suiji()
// 对随机数组排序 Arrays.sort(array)
System.out.println("产生的随机数组为: " + Arrays.toString(array))
System.out.println("要进行查找的值: ")
Scanner input = new Scanner(System.in)
// 进行查找的目标值 int aim = input.nextInt()
// 使用二分法查找 int index = binarySearch(array, aim)
System.out.println("查找的值的索引位置: " + index)
}
/** * 生成一个随机数组 *
* @return 返回值,返回一个随机数组 */
private static int[] suiji() {
// random.nextInt(n)+m 返回m到m+n-1之间的随机数 int n = new Random().nextInt(6) + 5
int[] array = new int[n]
// 循环遍历为数组赋值 for (int i = 0i <array.lengthi++) {
array[i] = new Random().nextInt(100)
}
return array
}
/** * 二分法查找 ---循环的方式实现 *
* @param array 要查找的数组 * @param aim 要查找的值 * @return 返回值,成功返回索引,失败返回-1 */
private static int binarySearch(int[] array, int aim) {
// 数组最小索引值 int left = 0
// 数组最大索引值 int right = array.length - 1
int mid
while (left <= right) {
mid = (left + right) / 2
// 若查找数值比中间值小,则以整个查找范围的前半部分作为新的查找范围 if (aim <array[mid]) {
right = mid - 1
// 若查找数值比中间值大,则以整个查找范围的后半部分作为新的查找范围 } else if (aim >array[mid]) {
left = mid + 1
// 若查找数据与中间元素值正好相等,则放回中间元素值的索引 } else {
return mid
}
}
return -1
}}
运行结果演示:
由以上运行结果我们得知,如果要查找的数据在数组中存在,则输出该数据在数组中的索引;如果不存在则输出 -1 ,也就是打印 -1 则该数在数组中不存在,反之则存在。
四、利用递归的方式实现二分法查找
public class BinarySearch2 {
public static void main(String[] args) {
// 生成一个随机数组 int[] array = suiji()
// 对随机数组排序 Arrays.sort(array)
System.out.println("产生的随机数组为: " + Arrays.toString(array))
System.out.println("要进行查找的值: ")
Scanner input = new Scanner(System.in)
// 进行查找的目标值 int aim = input.nextInt()
// 使用二分法查找 int index = binarySearch(array, aim, 0, array.length - 1)
System.out.println("查找的值的索引位置: " + index)
}
/** * 生成一个随机数组 * * @return 返回值,返回一个随机数组 */
private static int[] suiji() {
// Random.nextInt(n)+m 返回m到m+n-1之间的随机数 int n = new Random().nextInt(6) + 5
int[] array = new int[n]
// 循环遍历为数组赋值 for (int i = 0i <array.lengthi++) {
array[i] = new Random().nextInt(100)
}
return array
}
/** * 二分法查找 ---递归的方式 * * @param array 要查找的数组 * @param aim 要查找的值 * @param left 左边最小值 * @param right 右边最大值 * @return 返回值,成功返回索引,失败返回-1 */
private static int binarySearch(int[] array, int aim, int left, int right) {
if (aim <array[left] || aim >array[right]) {
return -1
}
// 找中间值 int mid = (left + right) / 2
if (array[mid] == aim) {
return mid
} else if (array[mid] >aim) {
//如果中间值大于要找的值则从左边一半继续递归 return binarySearch(array, aim, left, mid - 1)
} else {
//如果中间值小于要找的值则从右边一半继续递归 return binarySearch(array, aim, mid + 1, array.length-1)
}
}}
运行结果演示:
总结:
递归相较于循环,代码比较简洁,但是时间和空间消耗比较大,效率低。在实际的学习与工作中,根据情况选择使用。通常我们如果使用循环实现代码只要不是太繁琐都选择循环的方式实现~
我做了修改,现在没有问题了, package lianxiimport java.util.*public class SortFind { public static void main(String args[]) { int n = 0, low, high, middleSystem.out.println("从键盘输入一个整数,程序将判断该数是否在一个数组中")int a[] = { 12, 32, 9, -23, 45, 6, 46, 90, 123, 19, 34 }Arrays.sort(a)for (int i = 0i <a.lengthi++) { System.out.print(a[i] + " ")} Scanner reader = new Scanner(System.in)while (reader.hasNextInt()) { n = reader.nextInt()low = 0high = a.length - 1middle = (low + high) / 2while (low <= high) { System.out.println(middle)if (n == a[middle]) { System.out.println(n + "是数组中的元素")break} else if(n <a[0] || n >a[high]) { System.out.println(n + "不在数组中")break} else if (n <a[middle]) { high = middle - 1middle = middle-1} else if (n >a[middle]) { low = middle + 1middle = middle+1} } System.out.println("\n可继续输入整数,或输入非整数结束程序")} System.out.println("你输入的数据不是整数")} } 记得采纳#include <stdio.h>#define N 51
void main(void)
{
int a[N]
int i,n,num
int top,bottom,mid
int flag=1//如果在表列中找到数字,则值为1,否则为0
int loc=-1//要查找的数在表列中的位置,如果loca=-1表示表列中没有这个数如果有这个数,则它的值为所在的位置
printf("你想在多少个数中进行折半查找,请输入(1--50):")
scanf("%d",&n)
while(n<1 || n>50)
{
printf("你输入的数不正确,请重新输入。\n")
printf("你想在多少个数中进行折半查找,请输入(1--50):")
scanf("%d",&n)
}
printf("请你输入一个整数 a[1](需保证递增有序):")
scanf("%d",&a[1])
i=2
while(i<=n) //输入从小到大的表列
{
printf("请你输入一个整数 a[%d](需保证递增有序):",i)
scanf("%d",&a[i])
if(a[i] >a[i-1])
i++
else
printf("你输入的数不满足要求,请重新输入。\n")
}
//输出表列
printf("\n输出表列\n")
for(i=1i<=ni++)
{
printf("%6d",a[i])
}
printf("\n")
printf("请你输入要查找的数:")
scanf("%d",&num)
flag=1//假设输入的数在表列中
top=n
bottom=1
mid=(top+bottom)/2
while(flag)
{
//printf("top=%d, bottom=%d, mid=%d, a[%d]=%d\n",top,bottom,mid,mid,a[mid])
if( (num>a[top]) || (num<a[bottom]) ) //输入的数 num>a[top] 或者
num<a[bottom],肯定num不在这个表列中
{
loc=-1
flag=0
}
else if(a[mid]==num) //如果num 等于找到的数
{
loc=mid
printf("找到数 %6d 的位置%2d\n",num,loc)
break
}
else if(a[mid]>num) //若 a[mid]>num,则num 一定在 a[bottom]和a[mid-1]范围之内
{
top=mid-1
mid=(top+bottom)/2
}
else if(a[mid]<num) //若 a[mid]<num,则num 一定在 a[mid+1]和a[top]范围之内
{
bottom=mid+1
mid=(top+bottom)/2
}
}
if(loc==-1)
{
printf("%d 这个数在表列中没有找到。\n",num)
}
printf("折半查找结束,按任意键退出:\n")
}