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的时候剪枝

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

限制减法的次数

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

相关推荐
Savior`L6 小时前
二分算法及常见用法
数据结构·c++·算法
mmz12077 小时前
前缀和问题(c++)
c++·算法·图论
努力学算法的蒟蒻7 小时前
day27(12.7)——leetcode面试经典150
算法·leetcode·面试
甄心爱学习8 小时前
CSP认证 备考(python)
数据结构·python·算法·动态规划
kyle~8 小时前
排序---常用排序算法汇总
数据结构·算法·排序算法
AndrewHZ9 小时前
【遥感图像入门】DEM数据处理核心算法与Python实操指南
图像处理·python·算法·dem·高程数据·遥感图像·差值算法
CoderYanger9 小时前
动态规划算法-子序列问题(数组中不连续的一段):28.摆动序列
java·算法·leetcode·动态规划·1024程序员节
有时间要学习9 小时前
面试150——第二周
数据结构·算法·leetcode
liu****9 小时前
3.链表讲解
c语言·开发语言·数据结构·算法·链表