二分法中的两个模板

在acwing的算法基础课中,yxc给出了二分的两个模板,这里举有序数组查找某个数的例子来说明这两个模板。

模板1:

当我们将区间[l, r]划分成[l, mid]和[mid + 1, r]时,其更新操作是r = mid或者l = mid +1;,计算mid时不需要加1。
此操作用于check条件是获取右半部分的第一个元素。

int bsearch_1(int l, inr r){
    while(l<r){
        int mid = l+r>>1;
        if (check(mid)) r=mid;
        else l=mid+1;
    }
    return l;
}

模板2:

当我们将区间[l, r]划分成[l, mid - 1]和[mid, r]时,其更新操作是r = mid - 1或者l =mid;,此时为了防止死循环,计算mid时需要加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;
}

在这里插入图片描述
在这里插入图片描述

比如我们有一个有序数组,我们需要找到6的位置,我们可以将数组划分为两个部分,前半部分是小于6的部分,后半部分是大于等于6的部分,那么我们就有两种check的实现方式,分别对应取左半部分的最后一个<=6和取右半部分>=6的第一个:

  1. a[mid]>=6
    此时check函数对应第二部分,即原数组中>=6的部分。
    检查mid是否大于等于6,即我们需要获得第二部分中的第一个元素,其就是最终的答案。
    此时使用第一个模板
#include<iostream>
#include<vector>
using namespace std;

int main(){
	vector<int> a = {1,2,3,4,5,6,7,8,9};
	int l = 0;
	int r = a.size();
	while(l < r){
		int mid = (l + r)>>1;
		if(a[mid] >= 6) r = mid;
		else l = mid + 1;
	}
	cout << l;
	return 0;
} 
  1. a[mid]<=6
    此时check函数对应第一部分,即原数组中<=6的部分。
    检查mid是否小于等于6,即我们需要获得第一部分中的最后一个元素,其就是最终的答案。
    此时使用第二个模板
#include<iostream>
#include<vector>
using namespace std;

int main(){
	vector<int> a = {1,2,3,4,5,6,7,8,9};
	int l = 0;
	int r = a.size();
	while(l < r){
		int mid = (l + r + 1)>>1;
		if(a[mid] <= 6) l = mid;
		else r = mid - 1;
	}
	cout << l;
	return 0;
} 

为什么第二个模板需要l+r+1/2
因为如果r=l+1的时候,不+1的话mid就会等于l,如果进入check条件就会进入死循环。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

volcanical

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值