消失之物(背包DP)(容斥或分治)
容斥做法:
首先n^2搞出f[i][j]第i个物品,j体积的方案数。
去除每个物品贡献:
设个g[i][j]表示当i不选,j体积方案数(注意不是此时的范围相对于全局,而不是1---i)
那么我们用到一些容斥的思想
g[i][j]=f[n][j]-g[i][j-w[i]]
因为g[i][j-w[i]]即可表示当i选时的方案数(都是相对于全局的)
而且注意枚举顺序,我们要用当前g[i]的状态更新之后个g[i]的状态,
而且在j<w[i]的情况下g[i][j]=f[n][j]
注意取模.....
1 #include<map>
2 #include<iostream>
3 #include<cstdio>
4 #include<cstring>
5 #include<algorithm>
6 #include<cmath>
7 #include<string>
8 #include<vector>
9 #define MAXN 4101
10 #define int long long
11 using namespace std;
12 int f[MAXN][MAXN];
13 int g[MAXN][MAXN];
14 int n=0,w[MAXN];
15 int sum=0;
16 signed main()
17 {
18 scanf("%lld%lld",&n,&sum);
19 for(int i=1;i<=n;++i)
20 {
21 scanf("%lld",&w[i]);
22 }
23 f[0][0]=1;
24 for(int i=1;i<=n;++i)
25 {
26 for(int j=sum;j>=0;--j)
27 {
28 f[i][j]=f[i-1][j];
29 }
30 for(int j=sum;j>=w[i];--j)
31 {
32 f[i][j]=(f[i][j]+f[i-1][j-w[i]]+10)%10;
33 }
34 }
35 for(int i=1;i<=n;++i)
36 {
37 for(int j=0;j<=w[i]-1;++j)
38 {
39 g[i][j]=(f[n][j]+10)%10;
40 }
41 for(int j=w[i];j<=sum;++j)
42 {
43 g[i][j]=(f[n][j]-g[i][j-w[i]]+10)%10;
44 }
45 }
46 for(int i=1;i<=n;++i)
47 {
48 for(int j=1;j<=sum;++j)
49 {
50 printf("%lld",(g[i][j]+10)%10);
51 }
52 printf("\n");
53 }
54 }
分治:
%%%%skyh,非常强的思路
我们按照分治的做法,其实每一次我们只是不选其中的一件物品,其余不变,
那么我们利用这个性质,只用除了自身以外的物品更新当前答案
让我们想起树的结构,把叶子节点当作询问,
只需要每次递归左右区间,查询叶子节点时,左右都已经更新答案
回溯时重新更新
1 #include<iostream>
2 #include<cstdio>
3 #define int short
4 using namespace std;
5 const int N=2010;
6 int n,m,w[N],dp[15][N];
7 inline void add(int &a,int b){
8 a+=b;
9 if(a>=10) a-=10;
10 }
11 void solve(int dep,int l,int r){
12 if(l==r){
13 for(int i=1;i<=m;++i) printf("%hd",dp[dep-1][i]);
14 puts("");
15 return ;
16 }
17 int mid=l+r>>1;
18 for(int i=0;i<=m;++i) dp[dep][i]=dp[dep-1][i];
19 for(int i=mid+1;i<=r;++i) for(int j=m;j>=w[i];--j) add(dp[dep][j],dp[dep][j-w[i]]);
20 solve(dep+1,l,mid);
21 for(int i=0;i<=m;++i) dp[dep][i]=dp[dep-1][i];
22 for(int i=l;i<=mid;++i) for(int j=m;j>=w[i];--j) add(dp[dep][j],dp[dep][j-w[i]]);
23 solve(dep+1,mid+1,r);
24 }
25 signed main(){
26 scanf("%hd%hd",&n,&m);
27 for(int i=1;i<=n;++i) scanf("%hd",&w[i]);
28 dp[0][0]=1; solve(1,1,n);
29 return 0;
30 }
%%%skyh
消失之物(背包DP)(容斥或分治)的更多相关文章
- [Luogu P1450] [HAOI2008]硬币购物 背包DP+容斥
题面 传送门:https://www.luogu.org/problemnew/show/P1450 Solution 这是一道很有意思的在背包里面做容斥的题目. 首先,我们可以很轻松地想到暴力做背包 ...
- BZOJ 2287: 【POJ Challenge】消失之物( 背包dp )
虽然A掉了但是时间感人啊.... f( x, k ) 表示使用前 x 种填满容量为 k 的背包的方案数, g( x , k ) 表示使用后 x 种填满容量为 k 的背包的方案数. 丢了第 i 个, 要 ...
- 【bzoj2287】[POJ Challenge]消失之物 背包dp
题目描述 ftiasch 有 N 个物品, 体积分别是 W1, W2, ..., WN. 由于她的疏忽, 第 i 个物品丢失了. “要使用剩下的 N - 1 物品装满容积为 x 的背包,有几种方法呢? ...
- BZOJ2287: 【POJ Challenge】消失之物(背包dp)
题意 ftiasch 有 N 个物品, 体积分别是 W1, W2, ..., WN. 由于她的疏忽, 第 i 个物品丢失了. “要使用剩下的 N - 1 物品装满容积为 x 的背包,有几种方法呢?” ...
- bzoj 3622 DP + 容斥
LINK 题意:给出n,k,有a,b两种值,a和b间互相配对,求$a>b$的配对组数-b>a的配对组数恰好等于k的情况有多少种. 思路:粗看会想这是道容斥组合题,但关键在于如何得到每个a[ ...
- 【BZOJ 4665】 4665: 小w的喜糖 (DP+容斥)
4665: 小w的喜糖 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 94 Solved: 53 Description 废话不多说,反正小w要发喜 ...
- BZOJ 2287 【POJ Challenge】消失之物(DP+容斥)
2287: [POJ Challenge]消失之物 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 986 Solved: 572[Submit][S ...
- [LOJ#3119][Luogu5405][CTS2019]氪金手游(DP+容斥)
先考虑外向树的做法,显然一个点在其子树内第一个出现的概率等于它的权值除以它子树的权值和.于是f[i][j]表示i的子树的权值和为j时,i子树内所有数的相互顺序都满足条件的概率,转移直接做一个背包卷积即 ...
- HDU 5838 (状压DP+容斥)
Problem Mountain 题目大意 给定一张n*m的地图,由 . 和 X 组成.要求给每个点一个1~n*m的数字(每个点不同),使得编号为X的点小于其周围的点,编号为.的点至少大于一个其周围的 ...
随机推荐
- 更好的滚动体验>better-scroll
认识better-scroll better-scroll是一款重点用于解决移动端(已支持PC)各种滚动场景需求的插件,可使页面滚动效果更加流畅且富有弹性 better-scroll是用纯JavaSc ...
- select执行顺序
先from 找到表on过滤 找到两张表有对应关系的记录按join的方式添加外部行where 过滤group by分组having 过滤select 从having 过滤出来的字段中选择需要的字段dis ...
- 1.消息队列(queue)
版权声明:本文为博主原创文章,未经博主允许不得转载.https://www.cnblogs.com/Dana-gx/p/9724545.html 一.基本概念 IPC:Linux下的进程通信.包括6种 ...
- 【转载】有图 KVM折腾记..
KVM折腾记...https://lengjibo.github.io/KVM%E6%8A%98%E8%85%BE%E8%AE%B0/ Veröffentlicht am 2018-09-20 | ...
- Linux下使用bcwipe擦除磁盘空间
Linux下使用bcwipe擦除磁盘空间 2 Replies 如果要彻底删除硬盘上的文件,Windows下有磁盘粉碎机,bcwipe等. Linux下,也有bcwipe,而且功能更强大. 擦除磁盘剩余 ...
- python基础之迭代器、生成器、装饰器
一.列表生成式 a = [0,1,2,3,4,5,6,7,8,9] b = [] for i in a: b.append(i+1) print(b) a = b print(a) --------- ...
- 发现新大陆 --21lic
21lic网上发单平台 http://project.21ic.com/p/97250
- CyclicBarrier分析
简介 CyclicBarrier 是什么? CyclicBarrier 使用 CyclicBarrier 源码解析 CyclicBarrier 简单实现 barrierAction 是由哪个线程执行的 ...
- 韩小韩 API接口
官方网址:https://api.vvhan.com/ 天气API接口: https://api.vvhan.com/api/weather Bing每日图片API接口: https://api.vv ...
- 【原创】X86下ipipe接管中断/异常
目录 X86 ipipe接管中断/异常 一.回顾 二.X86 linux异常中断处理 1. 中断门及IDT 2. 初始化门描述符 2.1 早期异常处理 2.2 start_kernel中的异常向量初始 ...