原文链接http://www.cnblogs.com/zhouzhendong/p/8684027.html

题目传送门 - BZOJ2287

题意

  有$n$个物品,第$i$个物品的体积为$w_i$。

  令$cnt_{i,j}$表示不取第$i$个物品,占用$j$体积的方案总数。

  每一个物品只能取或者不取。

  让你对于每一个$i,j(1\leq i\leq n,1\leq j\leq m)$输出$cnt_{i,j}$。

  $n,m\leq 2\times 10^3$

题解

  这题有两种做法,时间复杂度不同,但是跑出来差不多,嘻嘻。

  $\Large Solution\ 1:$

  这个是经典的分治背包问题。第$i$个物品的出现时间为[1,i)U(i,n]。

  然后你会发现这个就是上一题BZOJ4025的弱化版。只是把并查集的一系列操作改成了$O(m)$背包DP而已。

  具体不再赘述,自行感受BZOJ4025的做法。

  时间复杂度$O(n^2\log n)$。

  $\Large Solution\ 2:$

  动态规划。

  首先处理出$f_n$表示没有任何限制搞01背包占用$n$体积的方案总数。

  考虑得到$cnt_{i,j}$。

  接下来分两种情况讨论。

  $j<w_i\Rightarrow cnt_{i,j}=f_j$:显然$f_j$的方案中不可能拿第$i$个物品啊。所以直接等于啊。

  $j\geq w_i\Rightarrow cnt_{i,j}=f_j-cnt_{i,j-w_i}$:考虑到$f_j$的方案中会有拿了第$i$个物品的情况,所以我们只要考虑减掉拿了第$i$个物品的情况。其他物品所占用的容量显然为$j-w_i$的。但是第$i$个物品只能拿一次啊,所以以后就不能拿了,所以是$cnt_{i,j-w_i}$。

  时间复杂度$O(n^2)$。

  然而,实际上跑出来差不多。

2679062 zhouzhendong 2287 Accepted 17008 kb 388 ms C++/Edit 610 B 2018-03-31 20:01:27
2679058 zhouzhendong 2287 Accepted 17004 kb 452 ms C++/Edit 942 B 2018-03-31 19:56:48

  下面的那个是分治的耗时,上面的那个是直接DP。

代码

分治

#include <bits/stdc++.h>
using namespace std;
const int N=2005;
vector <int> x,y;
int n,m,w[N],cnt[N][N];
void solve(int L,int R,vector <int> x,vector <int> &y){
vector <int> z;
z.clear();
for (int i=0;i<y.size();i++){
int id=y[i];
if (L<=id&&id<=R){
z.push_back(id);
continue;
}
for (int j=m-w[id];j>=0;j--)
x[j+w[id]]=(x[j+w[id]]+x[j])%10;
}
if (L==R){
for (int i=0;i<=m;i++)
cnt[L][i]=x[i];
return;
}
int mid=(L+R)>>1;
solve(L,mid,x,z);
solve(mid+1,R,x,z);
}
int main(){
scanf("%d%d",&n,&m);
for (int i=1;i<=n;i++)
scanf("%d",&w[i]);
x.clear(),y.clear();
x.push_back(1);
for (int i=1;i<=m;i++)
x.push_back(0);
for (int i=1;i<=n;i++)
y.push_back(i);
solve(1,n,x,y);
for (int i=1;i<=n;i++,puts(""))
for (int j=1;j<=m;j++)
printf("%d",cnt[i][j]);
return 0;
}

  

DP

#include <bits/stdc++.h>
using namespace std;
const int N=2005;
int n,m,w[N],f[N],cnt[N][N];
int main(){
scanf("%d%d",&n,&m);
for (int i=1;i<=n;i++)
scanf("%d",&w[i]);
memset(f,0,sizeof f);
f[0]=1;
for (int i=1;i<=n;i++)
for (int j=m-w[i];j>=0;j--)
f[j+w[i]]=(f[j+w[i]]+f[j])%10;
for (int i=1;i<=n;i++){
for (int j=0;j<w[i];j++)
cnt[i][j]=f[j];
for (int j=w[i];j<=m;j++)
cnt[i][j]=(f[j]-cnt[i][j-w[i]]+10)%10;
}
for (int i=1;i<=n;i++,puts(""))
for (int j=1;j<=m;j++)
printf("%d",cnt[i][j]);
return 0;
}

  

BZOJ2287 【POJ Challenge】消失之物 动态规划 分治的更多相关文章

  1. [bzoj2287][poj Challenge]消失之物_背包dp_容斥原理

    消失之物 bzoj-2287 Poj Challenge 题目大意:给定$n$个物品,第$i$个物品的权值为$W_i$.记$Count(x,i)$为第$i$个物品不允许使用的情况下拿到重量为$x$的方 ...

  2. bzoj2287 [POJ Challenge]消失之物

    题目链接 少打个else 调半天QAQ 重点在47行,比较妙 #include<algorithm> #include<iostream> #include<cstdli ...

  3. bzoj2287:[POJ Challenge]消失之物

    思路:首先先背包预处理出f[x]表示所有物品背出体积为x的方案数.然后统计答案,利用dp. C[i][j]表示不用物品i,组成体积j的方案数. 转移公式:C[i][j]=f[j]-C[i][j-w[i ...

  4. 【bzoj2287】[POJ Challenge]消失之物 背包dp

    题目描述 ftiasch 有 N 个物品, 体积分别是 W1, W2, ..., WN. 由于她的疏忽, 第 i 个物品丢失了. “要使用剩下的 N - 1 物品装满容积为 x 的背包,有几种方法呢? ...

  5. POJ Challenge消失之物

    Description ftiasch 有 N 个物品, 体积分别是 W1, W2, ..., WN. 由于她的疏忽, 第 i 个物品丢失了. "要使用剩下的 N - 1 物品装满容积为 x ...

  6. BZOJ.2287.[POJ Challenge]消失之物(退背包)

    BZOJ 洛谷 退背包.和原DP的递推一样,再减去一次递推就行了. f[i][j] = f[i-1][j-w[i]] + f[i-1][j] f[i-1][j] = f[i][j] - f[i-1][ ...

  7. 【bozj2287】【[POJ Challenge]消失之物】维护多值递推

    (上不了p站我要死了) Description ftiasch 有 N 个物品, 体积分别是 W1, W2, -, WN. 由于她的疏忽, 第 i 个物品丢失了. "要使用剩下的 N - 1 ...

  8. bzoj 2287: 【POJ Challenge】消失之物 动态规划

    Code: #include<cstdio> #include<algorithm> #include<queue> #include<cstring> ...

  9. 旧题再做【bzoj2287】【[pojchallenge]消失之物】分治背包

    (上不了p站我要死了) 今天听了 doggu神 讲了这道题的另一种做法,真是脑洞大开.眼界大开.虽然复杂度比黄学长的要大一点,但不总结一下简直对不起这神思路. 方法1:黄学长的做法(点这里) Desc ...

随机推荐

  1. 【Android开发经验】android:windowSoftInputMode属性具体解释

    本文章来自CSDN博客:http://blog.csdn.net/zhaokaiqiang1992.转载请注明地址! 在前面的一篇文章中,简单的介绍了一下怎样实现软键盘不自己主动弹出,使用的方法是设置 ...

  2. The servlet name already exists.解决方法

    The servlet name already exists.解决方法 当我们建立过同名的servlet文件,然后又将其删掉后再用同类名字建一个servlet时就会报错.解决办法:web.xml里面 ...

  3. 疯狂Workflow讲义——基于Activiti的工作流应用开 PDF 下载

    <疯狂Workflow讲义--基于Activiti的工作流应用开> 一:文档获取下载方式: 1:花20CSDN积分:可以下载:http://download.csdn.net/downlo ...

  4. Confluence 6 后台中的默认空间模板设置

    Confluence 6 后台中的默认空间模板设置界面的布局. https://www.cwiki.us/display/CONFLUENCEWIKI/Customizing+Default+Spac ...

  5. 团队开发工具git常用命令

    Git 常用命令 Git配置 git config --global user.name "storm" git config --global user.email " ...

  6. select前台转义后台取到的值为对应的文本 select同时接受list和map

    简单描述:select动态取值 要求是根据后台传过来的值在前台进行转义,emmm干就完了 思路分析:后台同时传过去一个map一个list ,map用来前台转义,list用来获取值,list取到的值相当 ...

  7. sass基础—属性嵌套以及跳出嵌套 @at-root

    /*注意:定义的变量若是没有使用则不会编译到css文件中.*//*1)sass的局部变量*/$font:14px;//定义$font:12px !default; //没有default时是重新赋值, ...

  8. ajax---异步请求对象的属性和方法

    方法: 1).open(method.url,asyn):创建请求,(post.get)asyn:表示同步(false)还是异步(true)提交 ,默认true 2)send(body) 发送请求,b ...

  9. MyBatis mysal 日报表,月,年报表的统计

    mysql 按日.周.月.年统计sql语句整理,实现报表统计可视化 原文地址:http://blog.csdn.net/u010543785/article/details/52354957 最近在做 ...

  10. SyntaxError: EOL while scanning string literal

    在Python 中,这个提示,一般是因为特殊字符引起的,比如换行符,比如 \ 等. 下面有几个示例: 1. 换行符 # 源错误代码 get_tabs="select b.owner,b.ta ...