BZOJ 4265 货币系统

时间:2022-09-03 18:01:49

今天比赛的时候做到的。题解写得很简单,但是感觉对于我这种蒟蒻还是很有思考的价值的。

题面(由于题面很短,就不概括了):小Q当上了新的宇宙大总统,他现在准备重新设计一套货币系统。

这个货币系统要求一共有m张货币,并且将他们按照币值从小到大排好序以后,前一个货币币值乘上x等于后一个货币币值,x∈{2,3,4,5},最小的币值为1。
小Q出门喜欢带总币值为n的钱,因为这是他的幸运数字,所以他希望设计出来的货币系统可以使得他带最少张数的货币。
数据范围:1≤n≤10^18 1≤k≤100

这道题(权限题),拿到的时候给人一种隐约的背包感。数据范围让小蒟蒻我想到DP+矩阵乘法/二分。然而这些都只是YY,我们仔细分析一下题面,可以先将其简(fu za)化为以下这个问题(ki∈{2,3,4,5},ai∈N*):
n=1*(a1+k1*(a2+k2*(a3+k3*(a4+k4*a5)))) 求出min(a1+a2+a3+a4+a5) (为了方便叙述,取m=5讨论)

这个还是比较好理解的。5个面值分别为1,k1,k1*k2,k1*k2*k3,k1*k2*k3*k4,5个面值各取a1,a2,a3,a4,a5张,把这个式子拆开就可以证明其正确性。然后我们可以得到,a2=(m-a1)/k2,后面的a3,a4,a5也是同理。

然后到这里小蒟蒻卡了一下。

反着考虑。假设我们已经得到所有货币的面值,肯定是从大到小贪心地取。因为 货币i 的一张等同于ki-1张 货币i-1 ,所以ai-1<ki-1 。因此,我们可以枚举ki,然后倒过来将n· 对ki的余数(也就是ai)计入答案,再将n·除以ki得到下一个n· ,直到除的次数(即k)用完。

因此,我们可以得到下面这个会T的算法(也就是小蒟蒻考试的时候上交的算法):

 #define ll long long
ll n;
int k;
ll dfs(ll xz,int cs){//xz表示当前值,cs表示还剩多少次
if(xz==)return ;
if(cs==)return xz;
ll mi=1e18,ans;
for(ll i=;i<=;i++){
ans=dfs(xz/i,cs-)+xz%i;
if(ans<mi)mi=ans;
}
return mi;
}
int main(){
scanf("%lld%d",&n,&k);
printf("%lld",dfs(n,k));
}

然后我就T了……

T了……

事实上,只要用map记忆化就可以了……怎么说呢……被自己蠢到了……

AC代码如下:

 #include <bits/stdc++.h>
using namespace std;
#define ll long long
ll n;
int k;
map<ll,ll>m[];
ll dfs(ll xz,int cs){
if(m[cs].count(xz))return m[cs][xz];
if(xz==)return ;
if(cs==)return xz;
ll mi=1e18,ans;
for(ll i=;i<=;i++){
ans=dfs(xz/i,cs-)+xz%i;
if(ans<mi)mi=ans;
}
return m[cs][xz]=mi;
}
int main(){
scanf("%lld%d",&n,&k);
printf("%lld",dfs(n,k));
}

以此篇博客为戒,多存套路,少犯智障错误。