题面:https://www.cnblogs.com/fu3638/p/6759919.html

硬币购物一共有4种硬币。面值分别为c1,c2,c3,c4。某人去商店买东西,去了tot次。每次带di枚ci硬币,买si的价值的东西,

  请问每次有多少种付款方法。其中di,s<=100000,tot<=1000。

  题解:
    首先考虑一个简单的问题,如果去掉题目中对于个数的限制,即给你四种面值的的硬币,问你有多少种方案能凑成
  si的价值。欸我们瞬间发现这是个完全背包的裸题,那果断乱搞。
    首先我们做一遍完全背包,定义f[i]为凑成i价值的方案数。
    接下来回到原题,我们发现题目加了一个di的限制,那怎么办呢(摸摸脑袋)。
    经过冷静的分析(查题解),发现这题可以用容斥原理乱搞。
    通过容斥原理,我们得出ans=全部方案(不考虑限制(即f[ans]))-Σ一种面值超过限制的方案数+Σ两种超限-Σ三种超限+Σ四种超限。
    那么如何求有几种超过限制的方案数呢
    以一种面值超过限制的方案数为例,那么这一种(不妨设为第i种)至少用d[i]+1个,即产生c[i]*(d[i]+1)的价值。那么剩下的s-c[i]*(d[i]+1)(记为rest)
  就可以随意取值,即为f[rest]种。
    那么两种三种的就是f[rest](rest=s-Σc[i]*(d[i]+1))。
    tip:枚举方案可以用位运算。
  代码:
#include<bits/stdc++.h>

using namespace std;

typedef long long ll;
const int maxs=;
ll ans,f[maxs+];
int c[],d[],tot,s; int main(){ scanf("%d%d%d%d%d",&c[],&c[],&c[],&c[],&tot); //完全背包预处理
f[]=;
for(int i=;i<=;i++)
for(int j=c[i];j<=maxs;j++)
f[j]+=f[j-c[i]]; for(int k=;k<=tot;k++){
scanf("%d%d%d%d%d",&d[],&d[],&d[],&d[],&s);
ans=;
for(int i=;i<(<<);i++){
int rest=s,tt=i,num=,pos=;
while(tt){
pos++;//第几枚硬币
if(tt&) rest-=c[pos]*(d[pos]+),num++;//num->几枚有限制
tt>>=;
}
if(rest<) continue;
if(num&) ans-=f[rest];
else ans+=f[rest];
}
printf("%lld\n",ans);
}
return ;
}

  参考:

    https://blog.csdn.net/aarongzk/article/details/51511564

     https://blog.csdn.net/doctor_godder/article/details/50071749


BZOJ 1042: [HAOI2008]硬币购物 (详解)(背包&容斥原理)的更多相关文章

  1. BZOJ 1042: [HAOI2008]硬币购物 容斥+背包

    1042: [HAOI2008]硬币购物 Description 硬币购物一共有4种硬币.面值分别为c1,c2,c3,c4.某人去商店买东西,去了tot次.每次带di枚ci硬币,买si的价值的东西.请 ...

  2. Bzoj 1042: [HAOI2008]硬币购物 容斥原理,动态规划,背包dp

    1042: [HAOI2008]硬币购物 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1747  Solved: 1015[Submit][Stat ...

  3. BZOJ 1042: [HAOI2008]硬币购物( 背包dp + 容斥原理 )

    先按完全背包做一次dp, dp(x)表示x元的东西有多少种方案, 然后再容斥一下. ---------------------------------------------------------- ...

  4. BZOJ 1042: [HAOI2008]硬币购物 [容斥原理]

    1042: [HAOI2008]硬币购物 题意:4种硬币.面值分别为c1,c2,c3,c4.1000次询问每种硬币di个,凑出\(s\le 10^5\)的方案数 完全背包方案数? 询问太多了 看了题解 ...

  5. bzoj 1042: [HAOI2008]硬币购物 dp+容斥原理

    题目链接 1042: [HAOI2008]硬币购物 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1706  Solved: 985[Submit][ ...

  6. BZOJ 1042 [HAOI2008]硬币购物(完全背包+容斥)

    题意: 4种硬币买价值为V的商品,每种硬币有numi个,问有多少种买法 1000次询问,numi<1e5 思路: 完全背包计算出没有numi限制下的买法, 然后答案为dp[V]-(s1+s2+s ...

  7. BZOJ 1042: [HAOI2008]硬币购物 容斥原理_背包_好题

    Description 硬币购物一共有4种硬币.面值分别为c1,c2,c3,c4.某人去商店买东西,去了tot次.每次带di枚ci硬币,买s i的价值的东西.请问每次有多少种付款方法. 题解: 十分喜 ...

  8. [BZOJ 1042] [HAOI2008] 硬币购物 【DP + 容斥】

    题目链接:BZOJ - 1042 题目分析 首先 Orz Hzwer ,代码题解都是看的他的 blog. 这道题首先使用DP预处理,先求出,在不考虑每种硬币个数的限制的情况下,每个钱数有多少种拼凑方案 ...

  9. [bzoj 1042][HAOI2008]硬币购物(用容斥原理弄背包)

    题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=1042 分析: 解法很巧妙,用f[i]表示四种硬币A.B.C.D的数量不考虑的情况下弄成 ...

随机推荐

  1. composer更改源为国际

    composer config -g repo.packagist composer https://repo.packagist.org

  2. 重构客户注册-基于ActiveMQ实现短信验证码生产者

    重构目标:将bos_fore项目中的CustomerAction作为短信消息生产者,将消息发给ActiveMQ,创建一个单独的SMS项目,作为短信息的消费者,从ActiveMQ获取短信消息,调用第三方 ...

  3. shell自定义输入输出 read+echo

    自定义格式输入.输出(244)  输出:echo -e 解释转义字符 -n  回车不换行 \n   新的一行,等同于回车 \t 制表符 \r 回车 \b 回退 baskspace 删除键 演示\n \ ...

  4. linux操作命令 开发人员需要掌握的一些命令

    1.man 查看帮助 2.命令 --help 简单帮助 3.help cd 查看一些Linux 命令行的一些内置命令 4.cp  粘贴复制命令  eg:cp yum.log /root/ 5.find ...

  5. Promise是什么?

    一. Promise是什么? Promise是一种异步操作的解决方案,将写法复杂的传统的回调函数和监听事件的异步操作,用同步代码的形式表达出来. 避免了多级异步操作的回调函数嵌套. Promise最早 ...

  6. mysql group by 对多个字段进行分组

    在平时的开发任务中我们经常会用到MYSQL的GROUP BY分组, 用来获取数据表中以分组字段为依据的统计数据.比如有一个学生选课表,表结构如下: Table: Subject_Selection S ...

  7. 搭建Hexo博客(四)-设置

    1.主题 比较流行的是yilia,下载主题到theme目录即可.再根据说明文档进行设置. 2.图片发布 source下创建的目录,并且目录下有文件,则文件会被生成到post目录中.图片可以放在这里.既 ...

  8. P2123 皇后游戏

    题目背景 还记得 NOIP 2012 提高组 Day1 的国王游戏吗?时光飞逝,光阴荏苒,两年 过去了.国王游戏早已过时,如今已被皇后游戏取代,请你来解决类似于国王游 戏的另一个问题. 题目描述 皇后 ...

  9. CentOS 安装、配置supervisord

    负责在启动自身时启动管理的子进程,响应客户端的命令,重启崩溃或退出的子进程,记录子进程stdout和stderr输出,生成和处理子进程生命周期中的事件. 安装yum install superviso ...

  10. linux shell系列10 判断某个月中的星期六和星期天

    #!/bin/bashread -p "请输入月份:" month #输入要查找的月份 mon=`date -d "0 month ago" +%m` #计算本 ...