【BZOJ5498】[十二省联考2019]皮配(动态规划)
【BZOJ5498】[十二省联考2019]皮配(动态规划)
题面
题解
先考虑暴力\(dp\),设\(f[i][j][k]\)表示前\(i\)所学校,有\(j\)人在某个阵营,有\(k\)人在某个派系的方案数。
发现如果\(k=0\),那么可以先决策每个城市选择哪一个阵营,再对于每个学校选择哪一个派系。显然两者之间不冲突,分开\(dp\)再乘起来就行了。
加入限制,每个限制的形式即在某个城市选定了某个阵营之后,这个学校只有一种选择。
先把没有限制的部分处理完,首先这些学校单独拎出来\(dp\)肯定没有问题。
不存在限制学校的城市也可以单独拎出来\(dp\)。
剩下的部分我们用前面的那个暴力\(dp\),这样子同时限制了两维就可以满足限制关系了。
最后我们枚举暴力\(dp\)的状态,利用前缀和就可以快速拼接两侧的答案。
#include<iostream>
#include<cstdio>
#include<vector>
using namespace std;
#define MOD 998244353
#define MAX 2550
inline int read()
{
int x=0;bool t=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=true,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return t?-x:x;
}
int n,m,C0,C1,D0,D1,ans;
int fr[MAX],S[MAX],ss[MAX],sum;
int g1[MAX],g2[MAX];
vector<int> p[MAX];
int K,lim[MAX],Lim[MAX];
int f[MAX][MAX],g[MAX][MAX];
int Calc(int x,int y)
{
int lc=max(0,sum-x-C1),rc=C0-x;
int ld=max(0,sum-y-D1),rd=D0-y;
if(lc>rc||ld>rd)return 0;
return 1ll*(g2[rc]-(lc?g2[lc-1]:0)+MOD)*(g1[rd]-(ld?g1[ld-1]:0)+MOD)%MOD;
}
int main()
{
int T=read();
while(T--)
{
n=read();m=read();C0=read();C1=read();D0=read();D1=read();
for(int i=1;i<=n;++i)
{
fr[i]=read();S[i]=read();
sum+=S[i];ss[fr[i]]+=S[i];
}
K=read();
for(int i=1;i<=n;++i)lim[i]=-1;
for(int i=1;i<=m;++i)Lim[i]=-1;
for(int i=1;i<=K;++i)
{
int x=read(),q=read();
lim[x]=Lim[fr[x]]=q;
p[fr[x]].push_back(x);
}
g1[0]=g2[0]=1;
int s1=0,s2=0;
for(int i=1;i<=n;++i)
{
s1+=S[i];if(~lim[i])continue;
for(int j=min(s1,D0);j>=S[i];--j)
g1[j]=(g1[j]+g1[j-S[i]])%MOD;
}
for(int i=1;i<=m;++i)
{
s2+=ss[i];if(!ss[i]||~Lim[i])continue;
for(int j=min(s2,C0);j>=ss[i];--j)
g2[j]=(g2[j]+g2[j-ss[i]])%MOD;
}
for(int i=1;i<=D0;++i)g1[i]=(g1[i-1]+g1[i])%MOD;
for(int i=1;i<=C0;++i)g2[i]=(g2[i-1]+g2[i])%MOD;
f[0][0]=1;
for(int i=1,sc=0,sd=0;i<=m;++i)
{
if(!~Lim[i]||!ss[i])continue;
for(int j=0;j<=C0&&j<=sc;++j)
for(int k=0;k<=D0&&k<=sd;++k)g[j][k]=f[j][k];
for(int qwq=0;qwq<(int)p[i].size();++qwq)
{
int x=p[i][qwq];sd+=S[x];
int t0=lim[x]!=0,t1=lim[x]!=1,t2=lim[x]!=2,t3=lim[x]!=3;
for(int j=min(C0,sc);~j;--j)
for(int k=min(D0,sd);~k;--k)
if(k>=S[x])
{
f[j][k]=(f[j][k]*t1+f[j][k-S[x]]*t0)%MOD;
g[j][k]=(g[j][k]*t3+g[j][k-S[x]]*t2)%MOD;
}
else f[j][k]=f[j][k]*t1,g[j][k]=g[j][k]*t3;
}
sc+=ss[i];
for(int j=min(C0,sc);~j;--j)
for(int k=min(D0,sd);~k;--k)
if(j>=ss[i])f[j][k]=(f[j-ss[i]][k]+g[j][k])%MOD;
else f[j][k]=g[j][k];
}
for(int i=0;i<=C0;++i)
for(int j=0;j<=D0;++j)
if(f[i][j])ans=(ans+1ll*f[i][j]*Calc(i,j))%MOD;
printf("%d\n",ans);
sum=ans=0;
for(int i=0;i<=D0;++i)g1[i]=0;
for(int i=0;i<=C0;++i)g2[i]=0;
for(int i=1;i<=n;++i)fr[i]=S[i]=0,lim[i]=0;
for(int i=1;i<=m;++i)p[i].clear(),ss[i]=0,Lim[i]=0;
for(int i=0;i<=C0;++i)for(int j=0;j<=D0;++j)g[i][j]=f[i][j]=0;
}
return 0;
}
【BZOJ5498】[十二省联考2019]皮配(动态规划)的更多相关文章
- luogu P5289 [十二省联考2019]皮配 背包
LINK:皮配 我承认是一道很难的题目. 不过对于这道题 部分分的提示显得尤为重要. 首先是 40分的暴力dp 很容易想 但是不容易写. 从40分可以发现我们只需要把蓝阵营和鸭派系的人数给存在起来就行 ...
- 洛谷P5289 [十二省联考2019]皮配(01背包)
啊啊啊边界判错了搞死我了QAQ 这题是一个想起来很休闲写起来很恶心的背包 对于\(k=0\)的情况,可以发现选阵营和选派系是独立的,对选城市选阵营和学校选派系分别跑一遍01背包就行了 对于\(k> ...
- Luogu5289 十二省联考2019皮配(动态规划)
将选择导师看成先选阵营再选派系,这样有显然的O(nm2)暴力,即按城市排序后,设f[i][j][k]为前i个学校中第一个阵营有j人第一个派系有k人的方案数,暴力背包. 对于k=0,可以发现选阵营和选派 ...
- luogu P5289 [十二省联考2019]皮配
传送门 首先考虑一个正常的dp,设\(f_{i,j,k}\)为前\(i\)个学校,\(j\)人在\(\color{#0000FF}{蓝阵营}\),\(k\)人在\(\color{#654321}{吔} ...
- 【LuoguP5289】[十二省联考2019] 皮配
题目链接 题目描述 略 Sol 一道背包问题 首先暴力做法设 \(dp[i][j][k]\) 表示前 \(i\) 个城市的学校被分到第一阵营 \(j\) 人 第一门派 \(k\) 人的方案数. 中间一 ...
- 【LOJ】#3051. 「十二省联考 2019」皮配
LOJ#3051. 「十二省联考 2019」皮配 当时我在考场上觉得这题很不可做... 当然,出了考场后再做,我还是没发现学校和城市是可以分开的,导致我还是不会 事实上,若一个城市投靠了某个阵营,学校 ...
- [十二省联考2019]字符串问题——后缀自动机+parent树优化建图+拓扑序DP+倍增
题目链接: [十二省联考2019]字符串问题 首先考虑最暴力的做法就是对于每个$B$串存一下它是哪些$A$串的前缀,然后按每组支配关系连边,做一遍拓扑序DP即可. 但即使忽略判断前缀的时间,光是连边的 ...
- [十二省联考2019]异或粽子——可持久化trie树+堆
题目链接: [十二省联考2019]异或粽子 求前$k$大异或区间,可以发现$k$比较小,我们考虑找出每个区间. 为了快速得到一个区间的异或和,将原序列做前缀异或和. 对于每个点作为右端点时,我们维护出 ...
- 【BZOJ5495】[十二省联考2019]异或粽子(主席树,贪心)
[BZOJ5495][十二省联考2019]异或粽子(主席树,贪心) 题面 BZOJ 洛谷 题解 这不是送分题吗... 转异或前缀和,构建可持久化\(Trie\). 然后拿一个堆维护每次的最大值,每次如 ...
随机推荐
- jupyter notebook安装、登录
pip install jupyter 提示pip需要升级(本人装的是anaconda) 输入:python -m pip install --upgrade pip 安装完成. 运行jupyter ...
- MongoDB在Linux下常用优化设置
MongoDB在Linux下常用优化设置 以下是一些MongoDB推荐的常用优化设置.在生产环境下选取合适的参数值,例如预读值和默认文件描述符数目等,会对系统性能有很大的影响. 1.关闭数据库文件的 ...
- Redis进阶学习笔记
Redis是什么这里不用再说了吧?下面是官方的解释 Redis is an open source (BSD licensed), in-memory data structure store, us ...
- win10的react native 开发环境搭建,使用Android模拟器
1.打开cmd的管理员模式,win+X,选择命令提示符(管理员)即可,运行如下命令: @"%SystemRoot%\System32\WindowsPowerShell\v1.0\power ...
- MySQL5.5.51启用网络远程连接
在其它电脑主机上访问时提示host ip is not allowed to connect to this mysql 下面代码为解决该问题的方法: :\Program Files\mysql-\b ...
- mysql的定时器
mysql定时器是系统给提供了event,而oracle里面的定时器是系统给提供的job.废话少说,下面创建表: create table mytable ( id int auto_incremen ...
- win10 家庭版不支持gpedit.msc的解决办法
win10 家庭版不支持gpedit.msc的解决办法 1.建立一个批处理文件内容如下: @echo off pushd "%~dp0" dir /b %systemroot%\W ...
- pycharm导入自定义py文件出错
1. 被导入的py文件不能以数字开头,否则会报错,红色波浪线
- 第一节 anaconda+jupyter+numpy简单使用
数据分析:是把隐藏在一些看似杂乱无章的数据背后的信息提炼出来,总结出所研究对象的内在规律 数据分析三剑客:Numpy,Pandas,Matplotlib 一 Anaconda 1 下载 官网:http ...
- DeconvNet 论文阅读理解
学习语义分割反卷积网络DeconvNet 一点想法:反卷积网络就是基于FCN改进了上采样层,用到了反池化和反卷积操作,参数量2亿多,非常大,segnet把两个全连接层去掉,效果也能很好,显著减少了参数 ...