二分专题2:寻找指定元素(模板)

首先先提一下二分模板:

二分,是 分两种情况,搜索的区间是哪一边。

下面模板代码中的判断性质函数:

bool check(int x){ /*  ...  */ }    // 检查 x 是否满足某种性质   x 为 v[mid]

第一种 情况   区间 [l,r] 分成   [l,mid]   和   [r,mid]  的时候:

模板代码如下:

// 区间[l,r] 被划分成 [l,mid] 和 [mid + 1,r]时使用:
int bsearch_1()
{
	// 这里我更新了别人的模板
	// 因为我们二分的时候需要注意好搜索区间
	// 我们模板用的是 while(l < r)  
	// 所以我们搜索的时候  r == n
	int l = 0,int r = n;
	
	while(l < r)
	{
		int mid = l + r >> 1;
		
		// 这里对应的是 我们的   [l,mid] 
		if(check(v[mid])) r = mid;		
		
		// 这里对应的是 我们的   [mid + 1,r]
		else l = mid + 1;
	}
	
	// 返回我们从左往右搜索的下标
	return l;
}

第二种 情况  区间[l,r]被划分成[l,mid - 1]和[mid,r]时使用:

模板代码如下:

// 区间[l,r] 被划分成 [l,mid - 1] 和 [mid,r]时使用:
int bsearch_2()
{
	// 这里我更新了别人的模板
	// 因为我们二分的时候需要注意好搜索区间
	// 我们模板用的是 while(l < r)  
	// 所以我们搜索的时候  r == n
	int l = 0,int r = n;
	
	while(l < r)
	{
		// 由于我们是二分搜索 右边的区域,
		// 所以我们要记得 r + 1  使得  mid 偏移向右边
		int mid = l + r + 1 >> 1;
		
		// 这里对应的是 我们的   [mid,r] 
		if(check(v[mid])) l = mid;		
		
		// 这里对应的是 我们的   [l,mid - 1]
		else l = mid - 1;
	}
	
	// 返回我们从左往右搜索的下标
	return l;
}

以上我的个人理解。

现在我们开始实践练习一下:

题目1:

 样例1:

输入
5 5
1 4 5 5 7

输出
2

 样例2:

输入
5 6
1 4 5 5 7

输出
4

 样例3:

输入
5 8
1 4 5 5 7

输出
5

解题:

        根据题目意思,我们利用函数模板套用即可,代码详解如下:

#include <iostream>
#define endl '\n'
#pragma GCC optimize(3,"Ofast","inline")
#define ___G std::ios::sync_with_stdio(false),cin.tie(0), cout.tie(0)
using namespace std;
const int N = 2e6 + 10;

int n,m,v[N];

inline int two_Find()
{

	// 虽然有二分查找 但是还是得注意
	// 控制好搜索区间
	int l = 0,r = n;

	// 由于我们是递增的顺序
	// 所以要找到 第一个大于等于 m 
	// 也就是  有效区间在   [l,mid] 
	// 我们要尽可能的插座左边多的元素,所以是  [l,mid]
	while(l < r)
	{
		int mid = l + r>> 1;
		if(v[mid] >= m) r = mid;
		else l = mid + 1;
	}

	// 返回左端点的下标,就是我们需要查找符合性质的元素
	return l;

}

inline void solve()
{
	cin >> n >> m;

	for(int i = 0;i < n;++i) cin >> v[i];

	cout << two_Find() << endl;

}


int main()
{
	___G;
	int _t = 1;
	// cin >> _t;
	while (_t--)
	{
		solve();
	}

	return 0;
}

 最后提交:

后面的题型基本上一致,套用模板即可。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值