二分答案


 基本模板 
在一个有序数组中二分查找一个值 k

 1 // 在一个单调数组里面二分查找某一个值 k
 2 int n, k;
 3 int a[MAXN];
 4
 5 void solve() {
 6     int lb = -1, ub = n;
 7     while (ub - lb > 1) {
 8         int mid = (lb + ub) / 2;
 9         if (a[mid] >= k) {
10             ub = mid;
11         } else {
12             lb = mid;
13         }
14     }
15     printf("%d\n", ub);
16 }

例题分析

POJ1064 

假定一个解并判断是否可行

题目:有 N 条绳子,长度分别为L。如果从他们中切割出 K 条长度相同的绳子的话,这 K 条绳子每条最长能有多长。答案保留到小数点后两位

解:C(x) = 可以得到 k 条长度为 x 的绳子 = floor((Li / x)的总和是否大于 k )

代码如下:

注意在程序判定中,我们的解的范围要求小数点后两位,因为 1 次循环可以将区间的范围缩小一半,100 次循环则可以达到 10 的精度范围。除此之外,也可以把中止条件设为像 (ub - lb)> EPS 这样,指定一个区间的大小

 1 int N, K;
 2 double L[MAXN];
 3
 4 bool C(double x) {
 5     int num = 0;
 6     for (int i = 0; i < N; i++) {
 7         num += (int)(L[i] / x);
 8     }
 9     return num >= K;
10 }
11
12 void solve() {
13     // 初始化解的范围
14     double lb = 0, ub = INF * 1.0;
15
16     // 重复循环,直到解的范围足够小
17     for (int i = 0; i < 100; i++) {
18         double mid = (lb + ub) / 2;
19         if (C(mid))
20             lb = mid;
21         else
22             ub = mid;
23     }
24     printf("%.2f\n", floor(ub * 100) / 100);
25 }
26
27 int main() {
28 #ifndef ONLINE_JUDGE
29     freopen("input.txt", "r", stdin);
30 #endif
31     scanf("%d %d", &N, &K);
32     for (int i = 0; i < N; i++) scanf("%lf", &L[i]);
33     solve();
34     return 0;
35 }

POJ 2456 

最大化最小值

题目:

农夫约翰打了一间有 N 间牛舍的小屋。牛舍排在一条线上,第 i 号牛舍在 x_i 的位置。但是他的 M 头牛对小屋很不满意,因此经常互相攻击。约翰为了防止牛之间的互相伤害,因此决定把每头牛都放在离其他牛尽可能远的牛舍。也就是要最大化最近的两头牛之间的距离。

解:

C(d)= 可以安排牛的位置使得最近的两头牛的距离不小于 d。则问题转化为求最大的 d

贪心:

对牛舍的位置进行排序;

把第一头牛放入 $x_0$ 的牛舍;

如果第 i 头牛放入了 $x_j$ 的话,第 $i+1$ 头牛就要放入满足 $x_j+d<=x_k$ 的最小的 $x_k$ 中

 1 int N, M;
 2 int x[MAXN];
 3
 4 bool C(int d) {
 5     int last = 1;
 6     for (int i = 1; i < M; i++) {
 7         int crt = last + 1;
 8         while (crt <= N && x[crt] - x[last] < d) {
 9             crt++;
10         }
11         if (crt == N + 1) return false;
12         last = crt;
13     }
14     return true;
15 }
16
17 void solve() {
18     sort(x + 1, x + 1 + N);
19     int lb = 0, ub = INF;
20     while (ub - lb > 1) {
21         int mid = (ub + lb) / 2;
22         if (C(mid))
23             lb = mid;
24         else
25             ub = mid;
26     }
27     printf("%d\n", lb);
28 }
29
30 int main() {
31 #ifndef ONLINE_JUDGE
32     freopen("input.txt", "r", stdin);
33 #endif
34     scanf("%d %d", &N, &M);
35     for (int i = 1; i <= N; i++) scanf("%d", &x[i]);
36     solve();
37     return 0;
38 }
01-15 02:03