洛谷P5289 [十二省联考2019]皮配(01背包)
啊啊啊边界判错了搞死我了QAQ
这题是一个想起来很休闲写起来很恶心的背包
对于\(k=0\)的情况,可以发现选阵营和选派系是独立的,对选城市选阵营和学校选派系分别跑一遍01背包就行了
对于\(k>0\)的情况,设\(f[i][0/1][j][k]\)表示对于第\(i\)个有限制的学校,该学校选择\(0/1\)阵营时,\(C0\)阵营有\(j\)人,\(D0\)派系有\(k\)人的方案数
转移要分类讨论,有点麻烦,看代码吧
// luogu-judger-enable-o2
#include <bits/stdc++.h>
#define N 1010
#define M 2505
#define R register
#define rep(i,x,y) for(i=x;i<=y;++i)
#define des(i,x,y) for(i=x;i>=y;--i)
#define mod 998244353
using namespace std;
int f[2][2][M][M],pref0[M],preg0[M],f0[M],g0[M];
int C0,C1,D0,D1,c0[N],tmp2[N],sum2=0;
int cnts[2],cntc[2],n,c;
bool visct[N];
inline int add(int x,int y){ return (x+y>=mod)?x+y-mod:x+y; }
inline int sub(int x,int y){ return (x-y<0)?x-y+mod:x-y; }
struct hhw{ int b,s,id,lim; } tmp[N],a0[N],ak[N];
inline void rd(int &x){
char c=getchar();R int y=0;
while(c<'0'||c>'9') c=getchar();
while(c>='0' && c<='9') y=y*10+c-'0',c=getchar();
x=y;
}
inline void solve0(){
R int i,j;
f0[0]=pref0[0]=1;
rep(i,1,cntc[0])
des(j,C0-c0[i],0)
f0[j+c0[i]]=add(f0[j+c0[i]],f0[j]);
rep(i,1,C0) pref0[i]=add(pref0[i-1],f0[i]);
g0[0]=preg0[0]=1;
rep(i,1,cnts[0])
des(j,D0-a0[i].s,0)
g0[j+a0[i].s]=add(g0[j+a0[i].s],g0[j]);
rep(i,1,D0) preg0[i]=add(preg0[i-1],g0[i]);
}
inline bool cmp(const hhw &x,const hhw &y){ return x.b<y.b; }
inline void addf(int a,int b,int c,int d,int x){
if(c>C0||d>D0) return;
f[a][b][c][d]=add(f[a][b][c][d],x);
}
inline void solvek(){
f[0][0][0][0]=1;
R int i,j,k,o,pos=1;
sort(ak+1,ak+cnts[1]+1,cmp);
rep(i,1,cnts[1]){
rep(o,0,1)
des(j,C0,0)
des(k,sum2,0){//不能直接写D0!!!要把有限制的学校的人数统计出来!!!
f[pos][o][j][k]=0;
if(o==0){
if(ak[i].lim^0){
if(ak[i].b==ak[i-1].b)
addf(pos,o,j,k+ak[i].s,f[pos^1][o][j][k]);
else{
addf(pos,o,j+tmp2[ak[i].b],k+ak[i].s,f[pos^1][o][j][k]);
addf(pos,o,j+tmp2[ak[i].b],k+ak[i].s,f[pos^1][o^1][j][k]);
}
}
if(ak[i].lim^1){
if(ak[i].b==ak[i-1].b)
addf(pos,o,j,k,f[pos^1][o][j][k]);
else{
addf(pos,o,j+tmp2[ak[i].b],k,f[pos^1][o][j][k]);
addf(pos,o,j+tmp2[ak[i].b],k,f[pos^1][o^1][j][k]);
}
}
} else{
if(ak[i].lim^2){
if(ak[i].b==ak[i-1].b)
addf(pos,o,j,k+ak[i].s,f[pos^1][o][j][k]);
else{
addf(pos,o,j,k+ak[i].s,f[pos^1][o][j][k]);
addf(pos,o,j,k+ak[i].s,f[pos^1][o^1][j][k]);
}
}
if(ak[i].lim^3){
if(ak[i].b==ak[i-1].b)
addf(pos,o,j,k,f[pos^1][o][j][k]);
else{
addf(pos,o,j,k,f[pos^1][o][j][k]);
addf(pos,o,j,k,f[pos^1][o^1][j][k]);
}
}
}
}
pos^=1;
}
}
inline int qwq(int x,int y){ return y>=0?sub(pref0[x],pref0[y]):pref0[x]; }
inline int owo(int x,int y){ return y>=0?sub(preg0[x],preg0[y]):preg0[x]; }
inline void solve(){
R int i,x,y,k,j,sum=0,ans=0;
rd(n),rd(c);
rd(C0),rd(C1),rd(D0),rd(D1);
rep(i,1,n){
rd(tmp[i].b),rd(tmp[i].s);
tmp[i].id=i,tmp[i].lim=-1;
sum+=tmp[i].s;
tmp2[tmp[i].b]+=tmp[i].s;
}
rd(k);
rep(i,1,k){
rd(x),rd(y),tmp[x].lim=y;
visct[tmp[x].b]=1;sum2+=tmp[x].s;
}
rep(i,1,n){
if(tmp[i].lim==-1) a0[++cnts[0]]=tmp[i];
else ak[++cnts[1]]=tmp[i];
}
rep(i,1,c){
if(!visct[i] && tmp2[i]) c0[++cntc[0]]=tmp2[i];
else cntc[1]++;
}
if(sum-C0>C1 || sum-D0>D1){
printf("0\n");
return;
}
solve0(),solvek();
rep(i,0,C0)
rep(j,0,D0){
if(sum-C1-i<=C0-i && sum-D1-j<=D0-j){
int x=add(f[cnts[1]&1][0][i][j],f[cnts[1]&1][1][i][j]);
ans=add(ans,1ll*x*qwq(C0-i,sum-C1-i-1)%mod*owo(D0-j,sum-D1-j-1)%mod);
}
f[0][0][i][j]=f[0][1][i][j]=f[1][0][i][j]=f[1][1][i][j]=0;
}
printf("%d\n",ans);
rep(i,1,c) tmp2[i]=visct[i]=0;
cntc[1]=cntc[0]=cnts[1]=cnts[0]=sum=sum2=0;
memset(f0,0,sizeof(int)*(C0+1));
memset(g0,0,sizeof(int)*(D0+1));
}
int main(){
R int t;rd(t);
while(t--) solve();
}
洛谷P5289 [十二省联考2019]皮配(01背包)的更多相关文章
- luogu P5289 [十二省联考2019]皮配
传送门 首先考虑一个正常的dp,设\(f_{i,j,k}\)为前\(i\)个学校,\(j\)人在\(\color{#0000FF}{蓝阵营}\),\(k\)人在\(\color{#654321}{吔} ...
- luogu P5289 [十二省联考2019]皮配 背包
LINK:皮配 我承认是一道很难的题目. 不过对于这道题 部分分的提示显得尤为重要. 首先是 40分的暴力dp 很容易想 但是不容易写. 从40分可以发现我们只需要把蓝阵营和鸭派系的人数给存在起来就行 ...
- 【BZOJ5498】[十二省联考2019]皮配(动态规划)
[BZOJ5498][十二省联考2019]皮配(动态规划) 题面 BZOJ 洛谷 题解 先考虑暴力\(dp\),设\(f[i][j][k]\)表示前\(i\)所学校,有\(j\)人在某个阵营,有\(k ...
- 洛谷.5284.[十二省联考2019]字符串问题(后缀自动机 拓扑 DP)
LOJ BZOJ 洛谷 对这题无话可说,确实比较...裸... 像dls说的拿拓扑和parent树一套就能出出来了... 另外表示BZOJ Rank1 tql... 暴力的话,由每个\(A_i\)向它 ...
- 洛谷.5283.[十二省联考2019]异或粽子(可持久化Trie 堆)
LOJ 洛谷 考场上都拍上了,8:50才发现我读错了题=-= 两天都读错题...醉惹... \(Solution1\) 先求一遍前缀异或和. 假设左端点是\(i\),那么我们要在\([i,n]\)中找 ...
- 洛谷P5284 [十二省联考2019]字符串问题 [后缀树]
传送门 思路 设\(dp_i\)表示以\(i\)结尾的\(A\)串,能达到的最长长度. 然后发现这显然可以\(i\)往自己控制的\(k\)连边,\(k\)往能匹配的\(j\)连边,就是个最长路,只要建 ...
- 洛谷 5291 [十二省联考2019]希望(52分)——思路+树形DP
题目:https://www.luogu.org/problemnew/show/P5291 考场上写了 16 分的.不过只得了 4 分. 对于一个救援范围,其中合法的点集也是一个连通块. 2n 枚举 ...
- 洛谷P5284 [十二省联考2019]字符串问题(SAM+倍增+最长路)
题面 传送门 题解 首先,我们把串反过来,那么前缀就变成后缀,建一个\(SAM\).我们发现一个节点的后缀是它的所有祖先 那么我们是不是直接按着\(parent\)树建边就可以了呢? 显然不是.我们假 ...
- Luogu5289 十二省联考2019皮配(动态规划)
将选择导师看成先选阵营再选派系,这样有显然的O(nm2)暴力,即按城市排序后,设f[i][j][k]为前i个学校中第一个阵营有j人第一个派系有k人的方案数,暴力背包. 对于k=0,可以发现选阵营和选派 ...
随机推荐
- C# 合并、拆分PPT幻灯片
概述 通过合并.拆分的功能,将不同的文档中的幻灯片进行组合形成新的的文档,同时也可以将一个多页的PPT文档按页拆分成多个不同的文档.此功能也丰富了编程人员对PPT幻灯片的操作的选择.下面将分别从以下几 ...
- 并发concurrent---2
背景:并发知识是一个程序员段位升级的体现,同样也是进入BAT的必经之路,有必要把并发知识重新梳理一遍. 并发concurrent: 使用ThreadLocal可以实现线程范围内共享变量,线程A写入的值 ...
- 对HTML5标签的认识(四)
这篇随笔讲讲HTML5中的表单和表单的一些元素 一.表单的作用是什么? 概念:表单在网页中主要是负责对数据信息的采取,表单一共分成三个部分: 1.表单的标签:这里面包含了处理表单的数据所用CGI程序以 ...
- 使用curl制作简易百度搜索
这几天研究了一下php中的curl类库,做了一个简单的百度搜索,先上代码 <div style="width:200px;height:100px;"> <div ...
- C++系列总结——多态
前言 封装隐藏了类内部细节,通过继承加虚函数的方式,我们还可以做到隐藏类之间的差异,这就是多态(运行时多态).多态意味一个接口有多种行为,今天就来说说C++的多态是怎么实现的. 编译时多态感觉没什么好 ...
- 你应该学会的Python多版本管理工具Pyenv
目录 Pyenv 简介 安装pyenv 通过pyenv安装python各种发行版 pyenv命令 多版本Python的管理 Pyenv常见问题Wiki Pyenv 简介 首先,该工具是在类linux环 ...
- JavaScript常用代码书写规范
javascript 代码规范 代码规范我们应该遵循古老的原则:“能做并不意味着应该做”. 全局命名空间污染 总是将代码包裹在一个立即的函数表达式里面,形成一个独立的模块. 不推荐 , y = ; c ...
- html文档知识补充
13.form表单(*******) 功能:前后数据交互,帮你提交任意的数据 input通过控制type属性来展示不同的获取用户输入的页面效果 type属性总结: text:纯文本 password: ...
- zList一个块状链表算法可以申请和释放同种对象指针,对于大数据量比直接new少需要差不多一半内存
zList是一个C++的块状内存链表,特点: 1.对于某种类别需要申请大量指针,zList是一个很好的帮手,它能比new少很多内存. 2.它对内存进行整体管理,可以将数据和文件快速互操作 3.和vec ...
- (七) Keras 绘制网络结构和cpu,gpu切换
视频学习来源 https://www.bilibili.com/video/av40787141?from=search&seid=17003307842787199553 笔记 首先安装py ...