Power Calculus(UVA 1374)

网址如下:

Power Calculus - UVA 1374 - Virtual Judge (vjudge.net)

用的迭代加深搜索做的

先上代码:

cpp 复制代码
#include<cstdio>
#include<cmath>

const int maxn = 2001;

bool vis[maxn];
int n, ans, arr[maxn];
int _2[100]{1};

int fetch2_(int cur){
    if(_2[cur]) return _2[cur];
    return _2[cur] = fetch2_(cur - 1) * 2;
}
void dfs(int d, int maxd, int maxm){
    if(arr[d - 1] == n) ans = maxd;
    if(d > maxd || maxm * fetch2_(maxd - d + 1) < n) return;

    //先加法延申
    for(int i = 0; i < d; i++){
        if(ans >= 0) return;
        int nxt = arr[d - 1] + arr[i];
        if(vis[nxt] || (maxm > n && nxt > n)) continue;
        arr[d] = nxt; vis[nxt] = true;
        dfs(d + 1, maxd, (maxm > nxt) ? maxm : nxt);
        vis[nxt] = false;
    }
    //后减法延申
    for(int i = 0; i < d; i++){
        if(ans >= 0) return;
        int nxt = arr[d - 1] - arr[i];
        if(nxt <= 0 || vis[nxt]) continue;
        arr[d] = nxt; vis[nxt] = true;
        dfs(d + 1, maxd, maxm);
        vis[nxt] = false;
    }
}

int main(void)
{
    vis[1] = true; arr[0] = 1;
    while(scanf("%d", &n) == 1 && n){
        ans = -1;
        for(int maxd = 0; ans == -1; maxd++)
            dfs(1, maxd, 1);
        printf("%d\n", ans);
    }

    return 0;
}

简单来说就是每一次"位移"都是选择两个数进行加法或除法来得到新的数,初始的数组只有1,代表的是x^1

列出几个优化方法:

不产生重复的数

只需要有一个数大于n

当最大的数乘2^(maxd - d)所得到的数小于n的时候剪枝

加法不是任选两个数,而是尽可能选更大的数

限制减法的次数

猜想:每次使用新得到的数

相关推荐
luthane2 小时前
python 实现average mean平均数算法
开发语言·python·算法
静心问道2 小时前
WGAN算法
深度学习·算法·机器学习
杰九2 小时前
【算法题】46. 全排列-力扣(LeetCode)
算法·leetcode·深度优先·剪枝
manba_2 小时前
leetcode-560. 和为 K 的子数组
数据结构·算法·leetcode
liuyang-neu2 小时前
力扣 11.盛最多水的容器
算法·leetcode·职场和发展
忍界英雄2 小时前
LeetCode:2398. 预算内的最多机器人数目 双指针+单调队列,时间复杂度O(n)
算法·leetcode·机器人
Kenneth風车2 小时前
【机器学习(五)】分类和回归任务-AdaBoost算法-Sentosa_DSML社区版
人工智能·算法·低代码·机器学习·数据分析
C7211BA3 小时前
使用knn算法对iris数据集进行分类
算法·分类·数据挖掘
Tisfy3 小时前
LeetCode 2398.预算内的最多机器人数目:滑动窗口+单调队列——思路清晰的一篇题解
算法·leetcode·机器人·题解·滑动窗口
程序猿练习生3 小时前
C++速通LeetCode简单第18题-杨辉三角(全网唯一递归法)
c++·算法·leetcode