二分法探微

摘抄自Acwing

二分模板一共有两个,分别适用于不同情况。
算法思路:假设目标值在闭区间 [ l , r ] [l, r] [l,r]中, 每次将区间长度缩小一半,当 l = r l = r l=r时,我们就找到了目标值。

模板一

当我们将区间 [ l , r ] [l, r] [l,r]划分成 [ l , m i d ] [l, mid] [l,mid] [ m i d + 1 , r ] [mid + 1, r] [mid+1,r]时,其更新操作是 r = m i d r = mid r=mid或者 l = m i d + 1 l = mid + 1 l=mid+1,计算 m i d mid mid时不需要加 1 1 1

int bsearch_1(int l, int r)
{
    while (l < r)
    {
        int mid = l + r >> 1;
        if (check(mid)) r = mid;
        else l = mid + 1;
    }
    return l;
}
  • 这里要注意的是 满足 c h e c k ( ) check\left(\right) check()条件的 m i d mid mid变量,此时被我们划分到了左区间中

模板二

当我们将区间 [ l , r ] [l, r] [l,r]划分成 [ l , m i d − 1 ] [l, mid - 1] [l,mid1] [ m i d , r ] [mid, r] [mid,r]时,其更新操作是 r = m i d − 1 r = mid - 1 r=mid1或者 l = m i d l = mid l=mid;,此时为了防止死循环,计算 m i d mid mid时需要加 1 1 1

int bsearch_2(int l, int r)
{
    while (l < r)
    {
        int mid = l + r + 1 >> 1;
        if (check(mid)) l = mid;
        else r = mid - 1;
    }
    return l;
}
  • 这里要注意的是 满足 c h e c k ( ) check\left(\right) check()条件的 m i d mid mid变量,此时被我们划分到了右区间中

这里面有一个问题就是为什么计算 m i d mid mid的时候还要加 1 1 1

其实,这一步最主要的目的是为了防止死循环。在这里我们举一个这样的例子

  1. 假设在程序的运算过程中的一步 l = 3 , r = 4 l=3,r=4 l=3,r=4,此时程序中的 m i d mid mid是这样计算的 m i d = ( l + r ) > > 1 mid=(l+r)>>1 mid=(l+r)>>1
  2. 如果在运算的一步中 c h e c k ( m i d ) check(mid) check(mid)为真。那么根据 m i d = l + r > > 1 mid=l+r>>1 mid=l+r>>1我们可以计算出 l = 3 l=3 l=3
  3. 然后,整个程序就陷入了死循环

m i d mid mid的计算关系到每一次运算过程结束后,区间是否能够经过充分的划分。

之所以会有这两种不同的模板,是因为在两种不同情况下, m i d mid mid会被分别分到左区间或者是右区间,我们必须保证每一种不同的划分方法都不能出现死循环。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值