B Antiamuny wants to leaern binary search again

题目:

C/C++:

int f(int l,int r,int x) { // l <= x <= r
    int cnt = 0;
    while(l <= r) {
        cnt++;
        int mid = (l + r) / 2;
        if (mid == x) break;
        if (mid < x) l = mid + 1;
        else r = mid - 1;
    }
    return cnt;
}

样例:

输入
5
3 7 2
6 12 2
2 10 3
6 14 8
5 8 1

输出
6
11
9
-1
6

思路:

        根据题意,暴力循环 l 到 r 一个样例肯定不会超时,但是这里有测试样例 t 就会导致超时,

刚好题目已经给出了二分模板,我们可以根据二分模板,二分查找 l 和 r 中是否有 x 的 cnt 循环满足我们需要找的 cnt 即可

代码详解如下:

#include <iostream>
#include <cstring>
#include <algorithm>
#include <queue>
#include <unordered_map>
#define endl '\n'
#define YES puts("YES")
#define NO puts("NO")
#define umap unordered_map
#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 cnt;	// 全局定义 cnt

// 题目所给的寻找 x 二分循环次数
inline int f(int l,int r,int x) { // l <= x <= r
    int cnt = 0;
    while(l <= r) {
        cnt++;
        int mid = (l + r) >> 1;
        if (mid == x) break;
        if (mid < x) l = mid + 1;
        else r = mid - 1;
    }
    return cnt;
}

// 这里我们二分对应的 l r 寻找 x 是否有我们要找的 cnt
inline int TwoFind(int l,int r) { // l <= x <= r
    // 这里的 tl 和 tr 是存储好原来的 l r,
	// 方便 我们 寻找二分 x cnt 的次数 
    int tl = l,tr = r,F;
    
	while(l <= r) {
    	
        int mid = (l + r) >> 1;
        // F 是二分 mid 后 cnt 次数
        F = f(tl,tr,mid);
        
        // 如果 当前的 mid 二分循环 cnt 次数与我们需要找的相符合
        // 则返回结果
		if (F == cnt) return mid;
        
        // 否则继续查找
		if (F < cnt) l = mid + 1;
        else r = mid - 1;
    }
    // 如果找不到,返回 -1
    return -1;
}

inline void solve()
{
	int l,r;
	cin >> l >> r >> cnt;
	int ans = TwoFind(l,r);
	cout << ans << endl;
}


int main()
{
//	freopen("a.txt", "r", stdin);
	___G;
	int _t = 1;
	cin >> _t;
	while (_t--)
	{
		solve();
	}

	return 0;
}

最后提交:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值