vijos-p1599 货币 -记忆化搜索

16 篇文章 0 订阅

描述

在某个神秘的星球上有一种货币,它有一种奇怪的兑换规则
你有且仅有一枚面值为n的金币,你可以去银行进行兑换,也可以不兑换
如果去银行兑换,兑换的规则是这样的:用面值为a的金币去兑换可以换到a/2,a/3,a/4这三枚硬币(如果
是小数则截尾取整),你可以兑换多次
读入n
输出你最后最多能拥有的钱数w
每个测试点中有T组数据

格式

输入格式

一个数T表示该点的测试数据组数(1=<T<=20 )
下面跟着T行,每行一个整数n(0 <= n <= 1000000000 )

输出格式

输出T行(一一对应)
每行一个整数就是你最后最多拥有的钱数w

样例1

样例输入1[复制]

2
12
2

样例输出1[复制]

13
2

限制

各个测试点3s

提示

小心数据较大,但是不需要高精度


解题思路

通过尝试发现,好几个不同的常数除以2、3、4后经常有重复的得数出现,所以将每个数能换取的最大价值存下来,这样就不用重复计算了。

代码

#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
//这里要注意都是long long
long long f[100000];//long long 占内存就大了,不能开太大的数组
long long ans;
long long dfs(long long n)
{
    int i;
    if (n <= 3) //计算发现前3都是自己本身,实际上前11都是自己本身
        return n;
     if(n<100000&&f[n]!=0)//注意数组越界!n很大
        return f[n]; // 有过的就不用搜了
    long long maxn=n,t=0;
    for(i=2;i<5;++i)
    {
        t+= dfs(n/i);
    }
    maxn=t > n ? t : n;
    if(n<100000)//数组越界
    {
        f[n]=maxn;
        return f[n];
    }
    else return maxn;


}

int main()
{
    long long t;
    cin>>t;//不知道为什么scanf用不了%lld,所以改成了cin

    while(t>0)
    {
        long long n;
        cin>>n;
        ans=dfs(n);
        cout<<ans<<endl;
        t--;
    }

    return 0;
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值