二分查找(折半查找)

题目:在一个有序数组中查找具体某个数字n

1、方法一:从前往后依次查找

#include <stdio.h>
int main()
{
    int arr[] = { 0,11,22,33,44,55,66,77,88,99 }; //假设的有序数列
    //元素位置:   0  1  2  3  4  5  6  7  8  9
    int k = 0; //k是要查找的数字
    scanf("%d", &k); //输入要查找的数字
    
    int i = 0;
    int sz = sizeof(arr) / sizeof(arr[0]);
    int flag = 0; //判断是否找到的标志
    for (i = 0; i < sz; i++)
    {
        if (k == arr[i])
        {
            flag = 1;
            printf("找到了,下标是:%d\n", i);
            break; //如果找到了这个数,便跳出循环
        }
    }
    if (flag == 0) //上面循环从前往后遍历了所有数组的值还没有找到,则说明找不到
        printf("找不到\n");
    return 0;
}

注:若数组很长,该做法效率低

2、方法二:二分查找(折半查找)

二分查找:通过将区间减半,逐渐缩小查找范围,从而快速完成查找

  • (1)首先我们将数组左端元素的下标定义为 left,最右端元素的下标定义为 right。根据左端元素的下标和右端元素的下标求出中间元素的下标 mid
  • (2)判断目标数字是在左侧还是右侧【k<arr[mid] — 左侧,k>arr[mid] — 右侧】;若【k==arr[mid]】,该数字即为要查找的数字
  • (3)若目标数字在 mid 左侧,把 mid-1 赋给 right;若在右侧,将 mid+1 赋给 left
  • (4)通过多次将区间折半,使得所需查找范围越来越小,从而加快查找效率
#include <stdio.h>
int main()
{
    int arr[] = { 0,11,22,33,44,55,66,77,88,99 }; //假设的有序数列
    //元素位置:   0  1  2  3  4  5  6  7  8  9
    int k = 0; //k是要查找的数字
    scanf("%d", &k); //输入要查找的数字
    
    int i = 0;
    int sz = sizeof(arr) / sizeof(arr[0]);
    int left = 0;
    int right = sz - 1;
    int flag = 0;
    while (left<=right) // = 不能忘掉
    {
        //int mid = (left + right) / 2;【可能导致整型越界问题】
        int mid = left + (right - left) / 2; //找到中间数
 
        if (arr[mid] < k) //这个数比中间数大
        {
            left = mid + 1; //则在右半查找
        }
        else if (arr[mid] > k) //这个数比中间数小
        {
            right = mid - 1; //则在左半查找
        }
        else
        {
            printf("找到了,下标是:%d\n", mid);
            flag = 1;
            break; //找到该数便跳出循环
        }
    }
    if (flag == 0)
        printf("找不到\n"); //上述循环查找完还没找到,说明此数不在给定的有序数列中
    return 0;
}

注:

  • (1)折半查找(二分查找),前提必须是数组有序
  • (2)第14行:while (left<=right) 中的 = 不能忘记加,因为可能出现两边同时向中间移动,到最后中间只剩一个元素,左右下标都指向该元素的情况
  • (3)第16行:int mid = (left + right) / 2; 该方法可行但还可以改进。因为如果考虑数组很长,left 和 right 都接近int型的最大值的话,两数加起来就超越了int型的规定范围,导致越界,这样的数除以2就不是他们两个的平均值了。故优化为【int mid = left + (right - left) / 2;】这样不会出现越界问题。
  • 13
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值