Bootstrap

39、【查找】二分查找:数的范围和查找某个数(C/C++版)

算法思想

二分查找(折半查找) 是针对有序序列的一种快速查找方式,与高中时候数学中学的二分法有异曲同工之处。通过不断缩小目标值所在区间的范围实现快速查找到目标值。每次将目标值与区间内的中位数进行比较,比中位数小则在中位数的做区间内再查找,如果比中位数大则在中位数的有区间内再查找,以相同的规则作用于缩小后的区间,以此类推最终便可找到结果。

对于整数二分
单调性(有序序列)仅为二分查找的充分条件,而不是必要条件,即具有单调性的序列一定可以进行二分查找,而可以进行二分查找的数列不一定具有单调性,非单调性的数列也有可能可以进行二分查找。

在这里插入图片描述
假设某个区间被划分成两个具有不同性质的子区间,整数二分的本质就是可以分别找到两个子区间的左右边界,从而将两个不同性质的子区间划分出来。

在这里插入图片描述
参考文章:
二分查找又叫折半查找,是一种简单又快速的查找算法

取mid值的两种方式:

(1)向下取整 (2)向上取整
在这里插入图片描述
因为二分查找是以中点来进行左右分割,因此类比于二叉树,可以将中点作为树根,左右两侧作为左右子树,用二叉树的形式来进行表示。
在这里插入图片描述

(1)向下取整
在这里插入图片描述
由于二分查找的表为顺序表,因此向下取整的意义就是取表中下标较靠前的位置,同时也是两个数中较小的那个数。

(2)向上取整
在这里插入图片描述

取表中下标较靠后的位置,同时也是两个数中较大的那个数。

算法实现注意事项

(1)当 while循环 判定条件为 l <= r

循环内改变mid时,
若target在中位点左侧时,则说明目标在左侧区间,改变右边界的取值,缩小搜索目标到左区间内。
r = mid - 1;

同理,若target在中位点的右侧时,则说明目标在右侧区间,
l = mid + 1;

l < = r所以左、右侧边界可以取到且可能会去到相同值。

(2)当 while循环 判定条件为 l < r (没有等号时)

在算法对整型数值进行二分查找中,主要是要考虑转换时的边界问题。

  1. 若选取的mid = (l + r) / 2 相当于向下取整,则需变换右边界时,应变为r = mid;当需变换左边界时,应变为l = mid + 1,最终将会得到靠左的位置l。
    在这里插入图片描述
    即可取到左侧边界,但不能取到右侧边界 [ ,)。

  2. 若选取的mid = (l + r + 1) / 2 相当于向上取整,则需变换左边界时,应变为l = mid;当需变换右边界时,应变为r = mid - 1,最终将会得到靠右的位置r。
    在这里插入图片描述
    即可渠道右侧边界,但不能取到左侧边界 ( , ]

算法模板

两步:获取中位数,和中位数对比

整型数二分

// l = 0, r = n
// x为带查找的数
int bsearch_1(int q[], int l, int r){	
	while(l < r){
		int mid = l + (r - l) / 2;
		// check():为判断q[mid]是否满足某种规定或性质
		//if(check(q[mid]))		r = mid;
		// 假设此时的目标是寻找x,在左侧区间
		if(q[mid] >= x)			r = mid;
		// 在右侧区间或等于q[mid] == x
		else					l = mid + 1;
	}
	return l;
}

// l = -1, r = n - 1
int bsearch_2(int q[], int l, int r){
	while(l < r){
		int mid = l + (r - l + 1) / 2;
		//if(check(q[mid]))		l = mid;
		if(q[mid] <= x)			l = mid;
		else					r = mid - 1;	
	}
	return r;
}

int bsearch_3(int q[], int l, int r){
	while (l <= r){
		int mid = l + (r - l) / 2;
		if(q[mid] <= x)		l = mid + 1;
		else				r = mid - 1;
	}
	return r;
}

注:为防止l + r相加溢出,采取l + (r - l) / 2 结果等于 (r - l) / 2

* 新模板(统一区间取值)
int left_bound(int target){
   int l = 0, r = n - 1;
   while(l <= r){
       int mid = l + (r - l) / 2;
       if(nums[mid] == target){	// 往左侧压缩,更新右边界
           r = mid - 1;
       }else if(nums[mid] > target){
           r = mid - 1;
       }else if(nums[mid] < target){
           l = mid + 1;
       }
   }

   if(l >= n || target != nums[l])  return -1;
   return l;	// 注意返回值

}

int right_bound(int target){
    int l = 0, r = n - 1;
    while(l <= r){
        int mid = l + (r - l) / 2;
        if(nums[mid] == target){ // 往右侧压缩,更新左边界
            l = mid + 1;
        }else if(nums[mid] > target){
            r = mid - 1;
        }else if(nums[mid] < target){
            l = mid + 1;
        }
    }

    if(r < 0 || target != nums[r])  return -1;
    return r;	// 注意返回值
}

浮点型二分

double bsearch(int q[], int l, int r){
	double eps = 1e-8;		// 精度,一般比题中要求的再多两位
	// l可为取值范围的左边界,r可为取值范围的有边界
	while(max((l - r), 1) > 1e-8){
		double mid = (l + r) / 2;
		// 若目标x小于mid,则在左区间找;否则在有区间找
		if(check(mid))		r = mid;
		else				l = mid;
	}
	return l;
}
  • 时间复杂度O(log n)
  • 空间复杂度O(1)

例题1:数的范围

给定一个按照升序排列的长度为 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

算法实现

#include <stdio.h>

const int N = 1e5 + 10;

int main(){
    int n, m;       scanf("%d%d", &n, &m);
    int q[N];       for(int i = 0; i < n; i++)      scanf("%d", &q[i]);
    
    while(m--){
        int k;      scanf("%d", &k);
        // 先获取左下标值。左闭右开。
        int i = 0, j = n - 1;
        while(i < j){
            int mid = i + j >> 1;
            if(k <= q[mid])         j = mid;
            else                    i = mid + 1;
        }
        if(k != q[j])        printf("-1 -1");
        else{
            printf("%d ", i);
            // 再获取右下标值。左开右闭。
            i = 0, j = n - 1;
            while(i < j){
                int mid = i + j + 1>> 1;
                if(k >= q[mid])     i = mid;
                else                j = mid - 1;
            }
            printf("%d", i);
        }
        //puts("");
        printf("\n");
    }
    return 0;
    
}

参考文章:
AcWing 789. 二分模板笔记

另一解法

#include <iostream>

using namespace std;

const int N = 1e5 + 10;

int n, q, k;
int nums[N];

int left_bound(int target){
   int l = 0, r = n - 1;
   while(l <= r){
       int mid = l + (r - l) / 2;
       if(nums[mid] == target){
           r = mid - 1;
       }else if(nums[mid] > target){
           r = mid - 1;
       }else if(nums[mid] < target){
           l = mid + 1;
       }
   }

   if(l >= n || target != nums[l])  return -1;
   return l;

}

int right_bound(int target){
    int l = 0, r = n - 1;
    while(l <= r){
        int mid = l + (r - l) / 2;
        if(nums[mid] == target){
            l = mid + 1;
        }else if(nums[mid] > target){
            r = mid - 1;
        }else if(nums[mid] < target){
            l = mid + 1;
        }
    }

    if(r < 0 || target != nums[r])  return -1;
    return r;
}

int main(){
    cin >> n >> q;
    int l, r;
    for(int i = 0; i < n; i++)      cin >> nums[i];
    while(q--){
        cin >> k;
        l = left_bound(k);
        if(l == -1)     cout << "-1 -1" << endl;
        else{
            r = right_bound(k);
            cout << l << " " << r << endl;
        }

    }

    return 0;
}


例题2:查找某个数

Leetcode:704. 二分查找

二分查找(C++、Python3)

「代码随想录」带你学透二分法!【704. 二分查找】

例题3:插入搜索位置

35. 搜索插入位置

注意返回值

例题4:在排序数组中查找元素的第一个和最后一个位置

34. 在排序数组中查找元素的第一个和最后一个位置

分左右边界讨论,寻找左边界时,向左侧区间压缩;寻找右边界时,向右侧区间压缩。

例题5:AcWing 790. 数的三次方根(浮点数二分)

AcWing 790. 数的三次方根(C++/Python)

;