IDA*
本页面将简要介绍 IDA*算法。
简介¶
IDA*为采用了迭代加深算法的 A*算法。由于 IDA*改成了深度优先的方式,相对于 A*算法,它的优势如下:
- 不需要判重,不需要排序;
- 空间需求减少。
伪代码¶
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 | Procedure IDA_STAR(StartState)
Begin
PathLimit := H(StartState) - 1;
Succes := False;
Repeat
inc(PathLimit);
StartState.g = 0;
Push(OpenStack, StartState);
Repeat
CurrentState := Pop(OpenStack);
If Solution(CurrentState) then
Success = True
Elseif PathLimit >= CurrentState.g + H(CurrentState) then
For each Child(CurrentState) do
Push(OpenStack, Child(CurrentState));
until Success or empty(OpenStack);
until Success or ResourceLimtsReached;
end;
|
优点¶
- 空间开销小:每个深度下实际上是一个深度优先搜索,不过深度有限制,使用 DFS 可以减小空间消耗;
- 利于深度剪枝。
缺点¶
重复搜索:即使前后两次搜索相差微小,回溯过程中每次深度变大都要再次从头搜索。
例题¶
埃及分数
在古埃及,人们使用单位分数的和(即
对于一个分数
输入整数
样例输入:
1 | 495 499
|
样例输出:
1 | Case 1: 495/499=1/2+1/5+1/6+1/8+1/3992+1/14970
|
解题思路
这道题目理论上可以用回溯法求解,但是解答树会非常“恐怖”——不仅深度没有明显的上界,而且加数的选择理论上也是无限的。换句话说,如果用宽度优先遍历,连一层都扩展不完,因为每一层都是 无限大 的。
解决方案是采用迭代加深搜索:从小到大枚举深度上限
深度上限
注意,这里使用 至少 一词表示估计都是乐观的。形式化地,设深度上限为
如果可以设计出一个乐观估价函数,预测从当前结点至少还需要扩展几层结点才有可能得到解,则迭代加深搜索变成了 IDA*算法。
示例代码
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 71 72 73 | // 埃及分数问题
#include <algorithm>
#include <cassert>
#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
int a, b, maxd;
long long gcd(long long a, long long b) { return b == 0 ? a : gcd(b, a % b); }
// 返回满足 1/c <= a/b 的最小 c 值
inline int get_first(long long a, long long b) { return b / a + 1; }
const int maxn = 100 + 5;
long long v[maxn], ans[maxn];
// 如果当前解 v 比目前最优解 ans 更优,更新 ans
bool better(int d) {
for (int i = d; i >= 0; i--)
if (v[i] != ans[i]) {
return ans[i] == -1 || v[i] < ans[i];
}
return false;
}
// 当前深度为 d,分母不能小于 from,分数之和恰好为 aa/bb
bool dfs(int d, int from, long long aa, long long bb) {
if (d == maxd) {
if (bb % aa) return false; // aa/bb 必须是埃及分数
v[d] = bb / aa;
if (better(d)) memcpy(ans, v, sizeof(long long) * (d + 1));
return true;
}
bool ok = false;
from = max(from, get_first(aa, bb)); // 枚举的起点
for (int i = from;; i++) {
// 剪枝:如果剩下的 maxd+1-d 个分数全部都是 1/i,加起来仍然不超过
// aa/bb,则无解
if (bb * (maxd + 1 - d) <= i * aa) break;
v[d] = i;
// 计算 aa/bb - 1/i,设结果为 a2/b2
long long b2 = bb * i;
long long a2 = aa * i - bb;
long long g = gcd(a2, b2); // 以便约分
if (dfs(d + 1, i + 1, a2 / g, b2 / g)) ok = true;
}
return ok;
}
int main() {
int kase = 0;
while (cin >> a >> b) {
int ok = 0;
for (maxd = 1; maxd <= 100; maxd++) { //枚举深度上限
memset(ans, -1, sizeof(ans));
if (dfs(0, get_first(a, b), a, b)) { //开始进行搜索
ok = 1;
break;
}
}
cout << "Case " << ++kase << ": ";
if (ok) {
cout << a << "/" << b << "=";
for (int i = 0; i < maxd; i++) cout << "1/" << ans[i] << "+";
cout << "1/" << ans[maxd] << "\n";
} else
cout << "No solution.\n";
}
return 0;
}
|
习题¶
build本页面最近更新:,更新历史
edit发现错误?想一起完善? 在 GitHub 上编辑此页!
people本页面贡献者:OI-wiki
copyright本页面的全部内容在 CC BY-SA 4.0 和 SATA 协议之条款下提供,附加条款亦可能应用