【C++】競程筆記(對答案二分搜)
===
[TOC]
程式碼範例參考:NTUCPC Guide,此筆記僅為個人學習用途。
---
例題:https://zerojudge.tw/ShowProblem?problemid=h084
暴力解:直接窮舉所有可能高度的 x,並且看海報寬度,遍歷木板高度選可以貼的地方,但是這樣時間複雜度會是 $O(n^2)$ ,木板高度可能高達 $10^9$ 。
具體程式碼寫法:
```cpp=
// 暴力破解法.cpp
#include <bits/stdc++.h>
using namespace std;
bool canPlacePosters(const vector<int>& h, const vector<int>& w, int x) {
int n = h.size();
int k = w.size();
int i = 0; // 木板的 index
int poster = 0;
while (poster < k) {
int needed = w[poster]; // 此張海報需要的寬度
// 在 h[i] >= x 的條件下,找長度為 w[poster] 的連續區間
int count = 0;
while (i < n) {
if (h[i] >= x) {
count++;
if (count == needed) {
i++; // 貼完這張後往下一格,避免重疊
break;
}
} else {
count = 0;
}
i++;
}
if (count < needed) return false;
poster++;
}
return true;
}
int main() {
int n, k;
cin >> n >> k;
vector<int> h(n);
for (int i = 0; i < n; ++i) cin >> h[i];
vector<int> w(k);
for (int i = 0; i < k; ++i) cin >> w[i];
int maxHeight = *max_element(h.begin(), h.end());
int ans = 0;
for (int x = 1; x <= maxHeight; ++x) {
if (canPlacePosters(h, w, x)) {
ans = x;
}
}
cout << ans;
return 0;
}
```
那這題可以用二分搜去解,若要用就需要找出 $x$ 的單調性。
> 先回去想想枚舉 $x$ 的過程,如果是從 $1$ 開始往上枚舉的話,其實在遇到第一個貼不完的高度時,就可以停下來了,畢竟高度越高的時候,能貼的地方只會越來越少,因此在更高的高度,也是不可能貼完的。當高度越高,用上面的方法能貼的海報就會越來越少、當高度越低,能貼的海報就越來越多。
> 這樣一來,我們就有一個「藏起來的序列」,這個序列的第 $x$ 項就是貼在高度 $x$ 的時候,最多可以貼幾張海報,長度就是我們本來的枚舉上界 $C = 10^9$ 。而這個序列是非嚴格遞減的,我們的目標是找到最後一個 $n$ 的位置,直接套一層二分搜就可以在 $O(n log C)$ 的時間解決這個問題了。
```cpp=
// 二分搜版本.cpp
// by LukeTseng
#include <bits/stdc++.h>
using namespace std;
bool canPlacePosters(const vector<int>& h, const vector<int>& w, int x) {
int n = h.size();
int k = w.size();
int i = 0, poster = 0;
while (poster < k) {
int need = w[poster];
int count = 0;
while (i < n) {
if (h[i] >= x) {
count++;
if (count == need) {
i++;
break;
}
} else {
count = 0;
}
i++;
}
if (count < need) return false;
poster++;
}
return true;
}
int main() {
int n, k;
cin >> n >> k;
vector<int> h(n);
for (int i = 0; i < n; ++i) cin >> h[i];
vector<int> w(k);
for (int i = 0; i < k; ++i) cin >> w[i];
int low = 1, high = *max_element(h.begin(), h.end());
int ans = 0;
while (low <= high) {
int mid = (low + high) / 2;
if (canPlacePosters(h, w, mid)) {
ans = mid;
low = mid + 1;
} else {
high = mid - 1;
}
}
cout << ans;
return 0;
}
```
習題
---
1. Zerojudge APCS 2017 年 3 月第 3 題 - 基地台:https://zerojudge.tw/ShowProblem?problemid=c575
題目說明:
* $N$ 個服務點座標 $P[0 \cdots N-1]$ 。
* 要架設 $K$ 個基地台,每個基地台可服務的範圍為一個區間,長度為 $D = 2R$ 。
* 目標是找出最小的整數 $D$ ,使得用 $K$ 個長度為 $D$ 的區間,能覆蓋所有服務點。
```
0 1 2 3 4 5 6 7 8 9
▲ ▲ ▲ ▲ ▲
```
假設 $K = 1$ ,則最小直徑為 $7$ ,只有一個的話就放在 $1$ 跟 $8$ 兩個服務點中間,也就是 $4.5$ (取中位數)。
而 $4.5$ 到 $1$ 的距離為 $|4.5 - 1| = 3.5$ ,這個就是半徑了,而另一邊也依此類推 $|4.5 - 8| = 3.5$ ,因此 $D = 3.5 \times 2 = 7$ 。
解題思路:
1. 排序服務點 $P[i]$ 。
2. 二分搜最小直徑 $D$ :由於題目保證最小直徑為整數,且 $D$ 至少為 $1$ ,最大可能為最大座標差(最大點 - 最小點),因此可在 $[1, max(P)-min(P)]$ 區間中對 $D$ 做**二分搜**。
3. 判斷是否能在這個 $D$ 下,用 $K$ 個基地台覆蓋所有點。
- 此為貪婪策略問題:
- 由左往右掃,遇到一個點沒被覆蓋,就以這個點作為左界,蓋一個基地台(長度為 $D$ )。
- 然後跳過所有在這個基地台範圍內的點(即從 $P[i]$ 開始,直到 $P[j] > P[i] + D$ 才停止)
- 重複此過程,每次都算一個新的基地台。
- 最後看總共用了幾個基地台 `used`,若 `used <= K`,則代表這個 $D$ 可行。
4. 若某個 $D$ 可行,則繼續去試更小的 $D$ (右界縮小),否則代表 $D$ 太小,左界往右擴。
註: $D$ 最大可能為最大座標差這件事,是從 $K = 1$ 去看的,因為你只有一個基地台可以架,又要涵蓋所有服務點,那 $D$ 不就最大座標 - 最小座標。
```cpp=
#include <bits/stdc++.h>
using namespace std;
// 判斷在直徑為 D 的情況下,是否能用 K 個基地台覆蓋所有服務點
bool check(const vector<int>& P, int K, int D) {
int N = P.size();
int count = 0; // 紀錄已使用的基地台數量
int i = 0; // 掃描目前位置
// 貪心法,從左至右覆蓋服務點
while (i < N) {
int cover_start = P[i]; // 當前基地台左邊緣為第 i 個服務點
int cover_end = cover_start + D; // 此基地台可覆蓋範圍為 [cover_start, cover_end]
count++; // 使用了一個基地台
// 繼續跳過所有在此基地台覆蓋範圍內的服務點
while (i < N && P[i] <= cover_end) {
i++;
}
}
// 如果所需基地台數量小於等於 K,代表此直徑可行
return count <= K;
}
int main() {
int N, K;
cin >> N >> K;
vector<int> P(N);
for (int i = 0; i < N; ++i) {
cin >> P[i];
}
sort(P.begin(), P.end());
// 找出最小與最大座標,用來設定二分搜左右界 l, r
int min_pos = *min_element(P.begin(), P.end());
int max_pos = *max_element(P.begin(), P.end());
int l = 1;
int r = max_pos - min_pos;
int ans = r; // 答案預設為最大直徑,之後再縮小
// 二分搜尋,找最小可行直徑
while (l <= r) {
int mid = (l + r) / 2;
if (check(P, K, mid)) {
ans = mid;
r = mid - 1;
} else {
l = mid + 1;
}
}
cout << ans;
return 0;
}
```
後面還有三題,但依我現在程度來說真的太難了XD,完全做不出來。