<—点个赞吧QwQ
宣传一下算法提高课整理
满足如下条件的序列 X(序列中元素被标号为 1、2、3…m)被称为“加成序列”:
- X[1]=1
- X[m]=n
- X[1]<X[2]<…<X[m−1]<X[m]
- 对于每个 k(2≤k≤m)都存在两个整数 i 和 j (1≤i,j≤k−1,i 和 j 可相等),使得 X[k]=X[i]+X[j]。
你的任务是:给定一个整数 n,找出符合上述条件的长度 m 最小的“加成序列”。
如果有多个满足要求的答案,只需要找出任意一个可行解。
输入格式
输入包含多组测试用例。
每组测试用例占据一行,包含一个整数 n。
当输入为单行的 0 时,表示输入结束。
输出格式
对于每个测试用例,输出一个满足需求的整数序列,数字之间用空格隔开。
每个输出占一行。
数据范围
1≤n≤100
输入样例:
5
7
12
15
77
0
输出样例:
1 2 4 5
1 2 4 6 7
1 2 4 8 12
1 2 4 5 10 15
1 2 4 8 9 17 34 68 77
思路
这道题是为迭代加深量身定制的。
迭代加深也叫IDDFS,它是类似BFS的算法,也是一层一层的扩展,只不过一个是DFS,一个是BFS。
那IDDFS到底是怎么一个算法呢?他其实是限定了DFS的搜索深度,这就是为了答案可能在很浅层但是DFS会搜太深以至于TLE。
我们从小到大枚举最大深度的大小,每次都这样寻找答案。
有的人可能就会问:那每次增加一层,不会重复计算很多次吗?
其实,就算搜索树是二叉的,多加的一层节点数仅比前面所有的节点少1!更何况k叉的搜索树!
回到这道题,这道题如果每次都以2n进行翻倍的话,那么序列是1,2,4,8,16,32,64,128,长度最多最多就只有8。但如果1,1,2,2,⋯,2永远也搜不到答案。所以这道题很适合IDDFS来做。
这里还可以加一个强剪枝:每一个数最多是前一个数的2倍,所以就可以提前判断是否无解。
代码
#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;
const int N = 10010;
int n;
int path[N];
bool dfs (int u,int depth) {
if (u > depth) return false;
if (path[u - 1] == n) return true;
if (path[u - 1] * (1ll << (depth - u + 1)) < n) return false;
bool st[N] = {};
for (int i = 0;i <= u - 1;i++) {
for (int j = 0;j <= i;j++) {
int sum = path[i] + path[j];
if (sum > n || sum <= path[u - 1] || st[sum]) continue;
st[sum] = true;
path[u] = sum;
if (dfs (u + 1,depth)) return true;
}
}
return false;
}
int main () {
path[0] = 1;
while (scanf ("%d",&n),n) {
int depth = 1;
while (!dfs (1,depth)) depth++;
for (int i = 0;i < depth;i++) printf ("%d%c",path[i],i == depth - 1 ? '\n' : ' ');
}
return 0;
}
佬迭代加深和BFS有什么区别
本质不同啊,迭代加深本质是 DFS
我知道了,bfs会爆内存