3233: [Ahoi2013]找硬币


Time Limit: 10 Sec  Memory Limit: 64 MB
Submit: 924  Solved: 482
[Submit][Status][Discuss]

Description


小蛇是金融部部长。最近她决定制造一系列新的货币。假设她要制造的货币的面值为x1,x2,x3… 那么x1必须为1,xb必须为xa的正整数倍(b>a)。例如 1,5,125,250就是一组合法的硬币序列,而1,5,100,125就不是。不知从哪一天开始,可爱的蛇爱上了一种萌物——兔纸!从此,小蛇便走上了遇上兔纸娃娃就买的不归路。某天,小蛇看到了N只可爱的兔纸,假设这N 只兔纸的价钱分别是a1,a2…aN。现在小蛇想知道,在哪一组合法的硬币序列下,买这N只兔纸所需要的硬币数最少。买兔纸时不能找零。
 
 
 

Input


第一行,一个整数N,表示兔纸的个数
第二行,N个用空格隔开的整数,分别为N只兔纸的价钱
 
 
 

Output


一行,一个整数,表示最少付的钱币数。

 
 
 

Sample Input


  

Sample Output


 

HINT


样例解释:共有两只兔纸,价钱分别为25和102。现在小蛇构造1,25,100这样一组硬币序列,那么付第一只兔纸只需要一个面值为25的硬币,第二只兔纸需要一个面值为100的硬币和两个面值为1的硬币,总共两只兔纸需要付4个硬币。这也是所有方案中最少所需要付的硬币数。

1<=N<=50, 1<=ai<=100,000

分析:


本来凑面值的题就是十分亲民的题,再加上数据又十分亲民,所以就是一道十分亲民的题……。

定义状态f[i]表示最大硬币为i面值时的最少硬币。转移方程f[i] = min(f[i],f[j] - sum((a[k] / i)* (i / j - 1)));(j为i的因子,k为每只兔子价格)

为什么这么转移,因为比如说我们有三枚三元硬币,我们可以转换成一枚九元硬币。

很简单,比如说我们有一只兔子价格为28,用1元硬币去凑会用28枚(即f[1] = 28),用1、3去凑会有  f[3] = f[1] - 28 / 3 * (3 / 1 - 1) = 10枚硬币,用1,3,9去凑会有f[9] = f[3] -  28 / 9 * (9 / 3 - 1) = 4枚硬币。

其实转移是很简单的,但是发现i的范围是100,000 ,j是i的因子需要去转化,k是每只兔子,复杂度很高的。我们可以考虑省掉j。即我们用递推。

可以由f[j]倒推f[i]。

发现100000 内每个数 不停加本身,推到100000,再乘上50只兔子的复杂度才几千万,随便卡过。

我一开始是这么做的,于是4.6S过了(数据太亲民了)。

其实后面我想了一下还可以优化,因为比如说f[27]由f[9]推过去肯定比由f[3]推过去优,所以对于每个数我们只用去推它的质数倍数就好了。

设M =100,000,N = 50;

K 为 M内所有数乘以质数倍推到100000的复杂度(好像这个数才十万左右....)

筛素数(M loglog M),递推(N *  K)所以复杂度是(M log log M + N * K)。然后就从4.6S降到了1.6S。话说数据是真亲民啊……

附上AC代码:


# include <iostream>
# include <cstdio>
# include <cstring>
# include <algorithm>
using namespace std;
const int N = ;
const int M = 1e5 + ;
int a[N],n;
int f[M],ans,sum,maxn,cnt,p[M];
bool vis[M];
void shai(){
for(int i = ;i < M;i++){
if(!vis[i])p[++cnt] = i;
for(int j = ;j <= cnt;j++){
if(p[j] * i >= M)break;
vis[p[j] * i] = true;
if(i % p[j] == )break;
}
}
}
void Init(){
memset(f,0x3f3f3f3f,sizeof f);
}
int dp(int i,int j){
int ans = f[j];
for(int k = ;k <= n;k++){
ans -= (a[k] / i) * (i / j - );
}
return ans;
}
int main(){
Init(); scanf("%d",&n);
for(int i = ;i <= n;i++){
scanf("%d",&a[i]);sum += a[i];
maxn = max(maxn,a[i]);
}
f[] = sum;ans = f[];
for(int i = ;i <= maxn;i++){
f[i] = min(f[i],dp(i,));
ans = min(ans,f[i]);
}
shai();
for(int i = ;i <= maxn;i++){
for(int j = ;j <= cnt;j++){
if(p[j] * i > maxn)break;
f[p[j] * i] = min(f[p[j] * i],dp(p[j] * i,i));
ans = min(ans,f[p[j] * i]);
}
}
printf("%d\n",ans);
}

[Bzoj3233][Ahoi2013]找硬币[基础DP]的更多相关文章

  1. BZOJ3233:[AHOI2013]找硬币(DP)

    Description 小蛇是金融部部长.最近她决定制造一系列新的货币.假设她要制造的货币的面值为x1,x2,x3… 那么x1必须为1,xb必须为xa的正整数倍(b>a).例如 1,5,125, ...

  2. [bzoj3233] [Ahoi2013]找硬币

    一开始没什么思路...后来想到确定最大硬币面值就知道其他面值能取多少了..而且结果是可以由较小的面值转移过来的. f[i]表示最大面值为i时的最小硬币数.a[i]表示第i个物品的价钱. f[i]=mi ...

  3. BZOJ 3233: [Ahoi2013]找硬币( dp )

    dp(x)表示最大面值为x时需要的最少硬币数. 枚举x的质因数p,  dp(x) = min( dp(x/p) - (p-1) * sigma[a[i]/x] ). ----------------- ...

  4. [AHOI2013]找硬币(搜索)

    [Ahoi2013]找硬币 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 348  Solved: 114[Submit][Status] Descri ...

  5. BZOJ 3233: [Ahoi2013]找硬币

    BZOJ 3233: [Ahoi2013]找硬币 标签(空格分隔): OI-BZOJ OI-DP Time Limit: 10 Sec Memory Limit: 64 MB Description ...

  6. 【bzoj 3233】[Ahoi2013]找硬币 ——搜索

    Description 小蛇是金融部部长.最近她决定制造一系列新的货币.假设她要制造的货币的面值为x1,x2,x3… 那么x1必须为1,xb必须为xa的正整数倍(b>a).例如 1,5,125, ...

  7. 【BZOJ 3233】 [Ahoi2013]找硬币

    [题目 描述] 小蛇是金融部部长. 最近她决定制造一系列新的货币. 假设她要制造的货币 的面值为 x1, x2, x3… 那么 x1 必须为 1, xb 必须为 xa 的正整数倍(b>a). 例 ...

  8. 「kuangbin带你飞」专题十二 基础DP

    layout: post title: 「kuangbin带你飞」专题十二 基础DP author: "luowentaoaa" catalog: true tags: mathj ...

  9. 找规律/数位DP HDOJ 4722 Good Numbers

    题目传送门 /* 找规律/数位DP:我做的时候差一点做出来了,只是不知道最后的 is_one () http://www.cnblogs.com/crazyapple/p/3315436.html 数 ...

随机推荐

  1. 修改JRE system library

    MyEclipse 默认的情况下JRE system library 是:MyEclipse 的,如何修改工程中的JRE system library呢?步骤如下: 1.选择工程->Proper ...

  2. 如何在Ubuntu里安装Helm

    Helm是什么?在战网上玩过暗黑破坏神2代的程序员们应该还记得,Helm是国度的意思. 而在计算机领域,Helm是什么? Helm是Kubernetes的一个包管理工具,有点像nodejs的npm,U ...

  3. 机器学习在SAP Cloud for Customer中的应用

    关于机器学习这个话题,我相信我这个公众号1500多位关注者里,一定有很多朋友的水平比Jerry高得多.如果您看过我以前两篇文章,您就会发现,我对机器学习仅仅停留在会使用API的层面上. 使用Java程 ...

  4. cluvfy comp命令用法

    1.获取集群验证工具cluvfy的帮助信息 grid@rac1:/home/grid>cluvfy -help USAGE: cluvfy [ -help ] cluvfy stage { -l ...

  5. billu_b0x靶场刷题

    https://www.vulnhub.com/ 里面有很多安全环境,只要下载相关镜像,在虚拟机上面搭建运行就可以练习对应靶场了. 第一步.信息收集 nmap扫描内网开放80端口的存活主机  nmap ...

  6. jquery--cookie应用

    示例:发送手机验证码 防止页面刷新后,发送验证码按钮重置 注:橙色部分为后增加代码,为防止验证码等待期间用户退出或者切换到其他页面以至于很久之后回到当前页面倒计时还在的问题,加入时间对比,记录用户发送 ...

  7. Navicat连不上MySQL的解决办法

    USE mysql; ALTER USER 'root'@'localhost' IDENTIFIED WITH mysql_native_password BY '{your password}'; ...

  8. 《发际线总是和我作队》第八次团队作业:Alpha冲刺 第五天

    项目 内容 这个作业属于哪个课程 软件工程 这个作业的要求在哪里 实验十二 团队作业8:软件测试与Alpha冲刺实验十一 团队作业7:团队项目设计完善&编码 团队名称 发际线总和我作队 作业学 ...

  9. POJ-1190-生日蛋糕(深搜,剪枝)

    生日蛋糕 Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 23049 Accepted: 8215 Description 7月1 ...

  10. 一直被用错的6种SQL 错误用法

    一直被用错的6种SQL 错误用法 1.LIMIT 语句 2.隐式转换 3.关联更新.删除 4.EXISTS语句 5.条件下推 6.提前缩小范围 sql语句的执行顺序: FROM ON JOIN WHE ...