二分查找算法
知识点
- 二分查找原理讲解
- 在单调递增序列
a
中查找 x
或 x
的后继
- 在单调递增序列
a
中查找 x
或 x
的前驱
二分查找算法讲解
枚举查找即顺序查找,实现原理是逐个比较数组 a[0:n-1]
中的元素,直到找到元素 x
或搜索整个数组后确定 x
不在其中。最坏情况下需要比较 N
次,时间复杂度是 O(n)
,属于线性阶算法。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| class Solution { public int search(int[] nums, int target) { int left = 0, right = nums.length; while (left < right) { int mid = left + ((right - left) >> 1); if (nums[mid] == target) return mid; else if (nums[mid] < target) left = mid + 1; else if (nums[mid] > target) right = mid - 1; } return -1; } }
|
折半查找的基本思想:
在有序表中(low, high, low<=high
),取中间记录即 a[(high+low)/2]
作为比较对象。
- 若给定值与中间记录的关键码相等,则查找成功。
- 若给定值小于中间记录的关键码,则在中间记录的左半区继续查找。
- 若给定值大于中间记录的关键码,则在中间记录的右半区继续查找。
不断重复上述过程,直到查找成功或所查找的区域无记录,查找失败。
二分查找的特征:
- 答案具有单调性。
- 二分答案的问题往往有固定的问法,例如:令最大值最小(最小值最大),求满足条件的最大(小)值等。
折半查找一般过程:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45
| Step 1:
假设存在一个有序数组: 下标[ 0 1 2 3 4 5 6 7 8 9 10 11 12 ] 数据[ 7 14 18 21 23 29 31 35 38 42 46 49 52 ] ↑ ↑ low=0 high=12
mid=(low+high)/2 mid=(0+12)/2 mid=6 [mid]=31 > 14,所以选择左半部分
操作: 此时令low不变,high=mid-1=5
Step 2:
下标[ 0 1 2 3 4 5 6 7 8 9 10 11 12 ] 数据[ 7 14 18 21 23 29 31 35 38 42 46 49 52 ] ↑ ↑ low=0 high=5
mid=(low+high)/2 mid=(0+6)/2 mid=3 [mid]=21 > 14,所以选择左半部分
操作: 此时令low不变,high=mid-1=2
Step 3:
下标[ 0 1 2 3 4 5 6 7 8 9 10 11 12 ] 数据[ 7 14 18 21 23 29 31 35 38 42 46 49 52 ] ↑ ↑ low=0 high=2
mid=(low+high)/2 mid=(0+2)/2 mid=1 [mid]=14 = 14 找到答案
操作: 返回下标
|
这个文本看起来更加清晰,修正了一些不规范的表达。
整数二分法常用算法模板
C++ 语言描述
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| while (low < high) { int mid = (low + high) / 2; if (a[mid] >= x) high = mid;
else low = mid + 1; }
while (low < high) { int mid = (low + high + 1) / 2;
if (a[mid] <= x) low = mid;
else high = mid - 1; }
|
Python 语言描述
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| while low<high:
mid=(low+high)/2
if(a[mid]>=x): high=mid
else: low=mid+1
while low<high:
mid=(low+high+1)/2
if(a[mid]<=x): low=mid
else: high = mid-1
|
Java 语言描述
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| while (low < high) {
int mid = (low + high) / 2;
if (a[mid] >= x) high= mid;
else low = mid + 1; }
while (low < high) {
int mid = (low + high + 1) / 2;
if (a[mid] <= x) low = mid;
else high = mid - 1;
}
|
此处我们先分整数的二分查找法的常用模版,关于实数的部分,我们后面再讲。
下面可能会有同学会疑问道:为什么采用这一套代码的而不是采用查找等于的 X?
是因为这样的适用范围更广,当有 X 时这套代码就返回 X 的位置。如果没有 X,就返回 <=x 的数中最大的一个或者 >=x 的数中最小的一个。
分巧克力
2017 年省赛真题链接。
题目描述: 儿童节那天有 K 位小朋友到小明家做客。小明拿出了珍藏的巧克力招待小朋友们。
小明一共有 $N$ 块巧克力,其中第 $i$ 块是 $H_i \times Wi$ 的方格组成的长方形。为了公平起见,
小明需要从这 $N$ 块巧克力中切出 K 块巧克力分给小朋友们。切出的巧克力需要满足:
- 形状是正方形,边长是整数;
- 大小相同;
例如一块 6x5 的巧克力可以切出 6 块 2x2 的巧克力或者 2 块 3x3 的巧克力。
当然小朋友们都希望得到的巧克力尽可能大,你能帮小明计算出最大的边长是多少么?
输入描述:
第一行包含两个整数 $N,K$ ($1 \leq N, K \leq 10^5$)。
以下 N 行每行包含两个整数 $H_i,W_i$ ($1 \leq H_i, W_i \leq 10^5$)。
输入保证每位小朋友至少能获得一块 1x1 的巧克力。
输出描述:
输出切出的正方形巧克力最大可能的边长。
输入输出样例:
示例:
输入
输出
运行限制:
注意:
- 请严格按要求输出,不要画蛇添足地打印类似:“请您输入…”的多余内容。
- 不要调用依赖于编译环境或操作系统的特殊函数。
- 所有依赖的函数必须明确地在源文件中
- 不能通过工程设置而省略常用头文件。
题目分析
简单思路,边长的最大规模为 $100000$;我们可以枚举出所有的情况。按从大到小的顺序进行切割,直到找到满足要求的巧克力边长。
在判断边长是否满足条件时:求一块长方形$(h * w)$最多被分成的正方形$(len * len)$巧克力个数为:
$cnt = (h / len) * (w / len)$
xxxxxxxxxx43 1import os2import sys34import heapq56a = []7b = []89n,k = map(int,input().split())1011for _ in range(n):12 x,y = map(int,input().split())13 a.append(x)14 b.append(y)151617q = []1819for i in range(n-1):20 heapq.heappush(q,(a[i],i))212223t = k2425ans = 02627while t > 0:28 w,i = heapq.heappop(q)2930 heapq.heappush(q,(b[i],i))31 ans += w32 t -= 1333435ans2 = 036if k >= n:37 ans2 += sum(a) + (k-n) * min(b)383940if k >= n:41 print(min(ans,ans2))42else:43 print(ans)python
即用在单调递增序列 $a$ 中查找 $<=x$ 的数中最大的一个(即 $x$ 或 $x$ 的前驱)即可,原本这里的条件是 $<=x$ ,我们将其换成验证即可。
代码解答
C++ 实现:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56
| #include<bits/stdc++.h>
using namespace std; const int MAXN=100010; int n,k; int h[MAXN],w[MAXN];
bool pd(int l) { int sum=0; for(int i=0; i<n; i++) { sum+=(h[i]/l)*(w[i]/l); if(sum>=k) { return true; } } return false; }
int main() { cin>>n>>k; for(int i=0; i<n; i++) cin>>h[i]>>w[i];
int high=0;
for(int i=0; i<n; i++) { high=max(high,h[i]); high=max(high,w[i]); } int low=1;
int mid=0; while(low<high) {
mid = (low + high+1) / 2; if(pd(mid)) low=mid; else high = mid - 1;
}
cout<<low; return 0; }
|
查找上界这里可以直接输入的时候查询,这道题实际上是可以少次操作的,代码如下。
Python 实现
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
| N=K=0 h=[] w=[] def pd(l):
sum1=0 for i in range(N):
sum1+= (h[i]//l)* (w[i]//l)
if sum1>=K : return True
return False
if __name__ == '__main__':
inFor = input().split()
N=int(inFor[0]) K=int(inFor[1])
high=0
for _ in range(N): wi, hi = map(int, input().split()) w.append(wi) h.append(hi) high=max(high,max((hi,wi)))
low=1 mid=0
while low<high :
mid= int ( (low+high+1)//2) if pd(mid): low=mid else: high=mid-1
print(low)
|
Java 实现
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51
| import java.util.Scanner; import static java.lang.Integer.max;
public class Main {
static int n, k; static int h[] = new int[100005]; static int w[] = new int[100005]; static boolean pd(int l) { int sum = 0; for (int i = 1; i <= n; i++) { sum += (h[i] / l) * (w[i] / l); if (sum >= k) { return true; } } return false; }
public static void main(String[] args) {
Scanner in = new Scanner(System.in); n = in.nextInt(); k = in.nextInt(); int high = 0;
for (int i = 1; i <= n; i++) { h[i] = in.nextInt(); w[i] = in.nextInt(); high = max(high, h[i]); high = max(high, w[i]); }
int low = 1; int mid = 0;
while (low < high) { mid = (low + high + 1) / 2; if (pd(mid)) low = mid; else high = mid - 1; } System.out.println(low); } }
|
模板中的 <= 和 => 都可以换成其他判定条件,像上面根据题目分析即可。
M 次方根
题目描述:
小 A 最近在学高等数学,他发现了一道题,求$\sqrt[3]{27}$ 。现在已知,小 A 开始计算,$1$ 的三次方得$1$,$2$ 的三次方得$8$,$3$ 的三次方得$27$,然后他很高兴的填上了$3$。
接着他要求$\sqrt[5]{164}$ 。然后他开始$1$ 的三次方得$1$,$2$ 的三次方得$8$,$3$ 的三次方得$27$…
直到他算到了秃头,也没有找到答案。
这时一旁的小 B 看不下去了,说这题答案又不是个整数。小 A 震惊,原来如此。作为程序高手的小 A,打算设计一个程序用于求解 $M$ 次根下$N$的值。
但是由于要考虑精度范围,答案必须要保留 $7$ 位小数,连三次根号下$27$都要掰手指的小 A 又怎么会设计呢。请你帮小 A 设计一个程序用于求解 $M$ 次根号$N$。
数据范围:
$1 \leq N \leq 1e5$ $1 \leq M \leq 100$ 且 $M < N$
要求输入:
1 2 3
| 输入描述:
第一行输入整数 N 和 M,数据间用空格隔开。
|
要求输出:
样例:
1 2 3 4 5 6 7
| 输入样例:
27 3
输出样例:
3.000000
|
运行限制:
1 2 3 4 5 6 7 8
| 最大运行时间:1s 最大运行内存: 256M
注意: 1. 请严格按要求输出,不要画蛇添足地打印类似:“请您输入…” 的多余内容。 2. 不要调用依赖于编译环境或操作系统的特殊函数。 3. 所有依赖的函数必须明确地在源文件中。 4. 不能通过工程设置而省略常用头文件。
|
题目分析
前面讲的都是整数二分,其实二分法还是可以用于实数。这个题目比较难,很多同学可能想不明白,想不明白就多读题,写写画画理解一下。这个题还有很多解法,现在告诉你了这道理用二分可以解答,请设计一个二分程序。
首先是这道题我们怎么下手:
根据前面的知识,我们要找到一个具有单调性的数列,去二分。这个题的关键是我们要去二分什么,这里可以二分的是 $a^M$ 中的 $a$,所以我们要先想办法设计出用于处理实数二分的代码。
这里给大家两个模板,都可以大家选择一个使用即可:
C++ 模版:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
|
while (l + eps < r) { double mid = (l + r) / 2;
if (pd(mid)) r = mid; else l = mid; }
for (int i = 0; i < 100; i++) {
double mid = (l + r) / 2; if (pd(mid)) r = mid; else l = mid; }
|
Python 模版:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
|
eps = 1e-6 while l + eps < r: mid = (l + r) / 2 if (pd(mid)): r = mid else: l = mid
for _ in range(100): mid = (l + r) / 2 if (pd(mid)): r = mid else: l = mid
|
Java 模版:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
|
while (l + eps < r) { double mid = (l + r) / 2;
if (pd(mid)) r = mid; else l = mid; }
for (int i = 0; i < 100; i++) {
double mid = (l + r) / 2; if (pd(mid)) r = mid; else l = mid; }
|
模板讲完了,然后我们就要考虑判定条件了,怎样判定是否存在满足大于平均值的区间。当然这个题你可以使用语言中自带开方软件,但是我们还是联系一下实数的二分代码。
关于判定条件,我们应该设计一个代码用于比较 $a^m$ 和 $N$ 的大小关系。
在我们代码中:
1 2 3 4
| if (pd(mid)) r = mid; else l = mid;
|
$pd$ 成功的情况,一定是 $pd$ 的 $mid$ 符合条件,且小于 $mid$ 的一定符合条件。因此我们要在大于 $mid$ 中继续查找,找到更大的 $mid$。
所以我们可以设计出如下判定条件:
1 2 3 4 5 6 7 8 9 10 11 12 13
| double pd(double a,int m) { double c=1; while(m>0) { c=c*a; m--; } if(c>=n) return true; else return false; }
|
代码解答
C++ 实现:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46
| #include <cstdio> #include <iostream> #include<iomanip> using namespace std;
double n,l,r,mid; double eps=1e-8;
bool pd(double a,int m) { double c=1; while(m>0) { c=c*a; m--; } if(c>=n) return true; else return false; }
int main() { int m; cin>>n>>m;
l=0,r=n;
while (l + eps < r) { double mid = (l + r) / 2; if (pd(mid,m)) r = mid; else l = mid; }
cout<<fixed<<setprecision(7)<<l; return 0; }
|
查找上界这里可以直接输入的时候查询,这道题实际上是可以少次操作的,代码如下。
Python 实现
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37
| n = 0.0 m = 0 l = 0.0 r = 0.0 mid = 0.0 eps = 0.00000001
def pd(a, m): c = 1.0 cnt = int(m)
while cnt > 0: c = c * a cnt -= 1
if c >= n: return True
else: return False
if __name__ == '__main__':
n, m = input().split()
l = 0 r=n=int(n)
while l + eps < r: mid = (l + r) / 2 if (pd(mid, m)): r = mid else: l = mid
print("%.7f" % l)
|
Java 实现
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70
| package com.company; import java.util.Scanner;
public class Main {
static double n, l, r, mid; static double eps = 1e-8;
static boolean pd(double a, int m) { double c = 1; while (m > 0) { c = c * a; m--; }
if (c >= n) return true; else return false; }
public static void main(String[] args) {
int m; Scanner in =new Scanner(System.in); n=in.nextDouble(); m=in.nextInt();
l = 0; r = n;
while (l + eps < r) { double mid = (l + r) / 2; if (pd(mid, m)) r = mid; else l = mid; } System.out.println(String.format("%.7f",l));
} }
|
总结
二分的题目主要是必须要求是单调的,一般会有条件等字眼。做这种题目主要还是找到递增或者递减的序列,然后关于序列的判定条件。或者通过观察时间复杂度来看是否可以使用二分,二分法的题目相对来说比较明显,设计起来也比较简单,模板不用死记硬背,理解一下,很快就可以独立写出来。