算法 - 二分

~~~~

  • [题目 - 整数二分需要考虑边界](#题目 - 整数二分需要考虑边界)
  • 思路
  • code
  • [开平方 - 浮点数二分](#开平方 - 浮点数二分)
  • code
  • [code core](#code core)

题目 - 整数二分需要考虑边界

给定一个按照升序排列的长度为 n 的整数数组,以及 q 个查询。

对于每个查询,返回一个元素 k 的起始位置和终止位置(位置从 0 开始计数)。

如果数组中不存在该元素,则返回 -1 -1。

输入格式

第一行包含整数 n 和 q,表示数组长度和询问个数。

第二行包含 n 个整数(均在 1∼10000 范围内),表示完整数组。

接下来 q 行,每行包含一个整数 k,表示一个询问元素。

输出格式

共 q 行,每行包含两个整数,表示所求元素的起始位置和终止位置。

如果数组中不存在该元素,则返回 -1 -1。

数据范围

1≤n≤100000

1≤q≤10000

1≤k≤10000

输入样例:

6 3

1 2 2 3 3 4

3

4

5

输出样例:

3 4

5 5

-1 -1


思路

      1. 升序整数数组,查找一个数x在该数组中的范围l,r,依据数x可以把数组分为两部分:大于等于x的部分,小于x的部分。先找左边界,再找右边界。
      1. 左边界:x的左边都小于x,右边都大于等于x;中间的数mid与x比较,mid >= x,说明x在l,mid范围,包含mid,因为mid可能是x;反之mid < x,说明x在mid + 1, r范围内,不包含mid,因为mid一定小于x。

右边届:x的左边都小于等于x,右边的数都大于x;中间的数mid与x比较,mid <= x,说明x在mid, r范围,包含mid,因为mid可能是x;反之mid > x,说明x在l, mid - 1范围内,不包含mid,因为mid一定大于x。

      1. 注意求mid时,一个是l + r >> 1;一个是l + r + 1 >> 1;这个+1是为了防止l = r - 1时即l与r相邻时导致的死循环问题。

code

cpp 复制代码
#include <iostream>

using namespace std;

const int N = 100010;

int q[N];

int n, m;

int main(){
    scanf("%d %d", &n, &m);
    
    for(int i = 0; i < n; i ++) scanf("%d", &q[i]);
    
    while(m --){
        int x;
        scanf("%d", &x);
        
        int l = 0, r = n - 1;
        
        while(l < r){
            int mid = l + r >> 1;
            if(q[mid] >= x) r = mid;
            else l = mid + 1;
        }
        
        if(q[l] != x) cout << "-1 -1" << endl;
        else{
            cout << l << " ";
            
            l = 0, r = n - 1;
            
            while(l < r){
                int mid = l + r + 1 >> 1;
                if(q[mid] <= x) l = mid;
                else r = mid - 1;
            }
            cout << r << endl;
        }
        
    }
    return 0;
}

开平方 - 浮点数二分

求一个数的平方根

code

cpp 复制代码
#include <iostream>

using namespace std;

int main(){
	double x;
	cin >> x;
	double l = 0, r = x;
	while(r - l > 1e-6){//for(int i = 0; i < 100; i ++) 循环100次也行
		double mid = (l + r) / 2;
		if(mid * mid >= x) r = mid;
		else l = mid;
	}
	printf("%lf\n", l);
	return 0;
}

code core

cpp 复制代码
bool check(int x) {/* ... */} // 检查x是否满足某种性质

// 区间[l, r]被划分成[l, mid]和[mid + 1, r]时使用:
int bsearch_1(int l, int r)
{
    while (l < r)
    {
        int mid = l + r >> 1;
        if (check(mid)) r = mid;    // check()判断mid是否满足性质
        else l = mid + 1;
    }
    return l;
}
// 区间[l, r]被划分成[l, mid - 1]和[mid, r]时使用:
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;
}

E N D END END

相关推荐
复杂网络4 小时前
多个 Claude Code 与多个 Codex 协同工作:设计与实现方案
算法
HjhIron19 小时前
面试常客:字符串算法从入门到进阶
算法·面试
吴佳浩20 小时前
DeepSeek DSpark:Confidence-Scheduled Speculative Decoding 技术解析
人工智能·算法·deepseek
触底反弹1 天前
🧠 搞懂 Token,才算真正入门大模型——从分词原理到 Embedding 语义实战
javascript·人工智能·算法
vivo互联网技术1 天前
ICLR 2026 | 基于后验采样的图像恢复方法LearnIR:人脸去阴影、去雾
人工智能·算法·aigc
浮生望1 天前
JS字符串与回文算法:从包装类到双指针的面试进阶之路
javascript·算法
黄敬峰1 天前
面试必刷:从JS底层包装类到双指针,彻底搞懂字符串与回文算法
算法
地平线开发者2 天前
J6B vio scenario sample
算法
BothSavage2 天前
Trae远程开发中DeepSeek自定义模型4054错误的排查与修复
算法