【BZOJ5498】[十二省联考2019]皮配(动态规划)

题面

BZOJ

洛谷

题解

先考虑暴力\(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]皮配(动态规划)的更多相关文章

  1. luogu P5289 [十二省联考2019]皮配 背包

    LINK:皮配 我承认是一道很难的题目. 不过对于这道题 部分分的提示显得尤为重要. 首先是 40分的暴力dp 很容易想 但是不容易写. 从40分可以发现我们只需要把蓝阵营和鸭派系的人数给存在起来就行 ...

  2. 洛谷P5289 [十二省联考2019]皮配(01背包)

    啊啊啊边界判错了搞死我了QAQ 这题是一个想起来很休闲写起来很恶心的背包 对于\(k=0\)的情况,可以发现选阵营和选派系是独立的,对选城市选阵营和学校选派系分别跑一遍01背包就行了 对于\(k> ...

  3. Luogu5289 十二省联考2019皮配(动态规划)

    将选择导师看成先选阵营再选派系,这样有显然的O(nm2)暴力,即按城市排序后,设f[i][j][k]为前i个学校中第一个阵营有j人第一个派系有k人的方案数,暴力背包. 对于k=0,可以发现选阵营和选派 ...

  4. luogu P5289 [十二省联考2019]皮配

    传送门 首先考虑一个正常的dp,设\(f_{i,j,k}\)为前\(i\)个学校,\(j\)人在\(\color{#0000FF}{蓝阵营}\),\(k\)人在\(\color{#654321}{吔} ...

  5. 【LuoguP5289】[十二省联考2019] 皮配

    题目链接 题目描述 略 Sol 一道背包问题 首先暴力做法设 \(dp[i][j][k]\) 表示前 \(i\) 个城市的学校被分到第一阵营 \(j\) 人 第一门派 \(k\) 人的方案数. 中间一 ...

  6. 【LOJ】#3051. 「十二省联考 2019」皮配

    LOJ#3051. 「十二省联考 2019」皮配 当时我在考场上觉得这题很不可做... 当然,出了考场后再做,我还是没发现学校和城市是可以分开的,导致我还是不会 事实上,若一个城市投靠了某个阵营,学校 ...

  7. [十二省联考2019]字符串问题——后缀自动机+parent树优化建图+拓扑序DP+倍增

    题目链接: [十二省联考2019]字符串问题 首先考虑最暴力的做法就是对于每个$B$串存一下它是哪些$A$串的前缀,然后按每组支配关系连边,做一遍拓扑序DP即可. 但即使忽略判断前缀的时间,光是连边的 ...

  8. [十二省联考2019]异或粽子——可持久化trie树+堆

    题目链接: [十二省联考2019]异或粽子 求前$k$大异或区间,可以发现$k$比较小,我们考虑找出每个区间. 为了快速得到一个区间的异或和,将原序列做前缀异或和. 对于每个点作为右端点时,我们维护出 ...

  9. 【BZOJ5495】[十二省联考2019]异或粽子(主席树,贪心)

    [BZOJ5495][十二省联考2019]异或粽子(主席树,贪心) 题面 BZOJ 洛谷 题解 这不是送分题吗... 转异或前缀和,构建可持久化\(Trie\). 然后拿一个堆维护每次的最大值,每次如 ...

随机推荐

  1. SuperMap -WebGL 实现地球的背景透明并显示自定义图片

    实现效果如图: 实现代码如下: <!DOCTYPE html> <html lang="en"> <head> <meta charset ...

  2. Salesforce 小知识:大量“子记录”的处理方法

    大量"子记录"的存放 例子:系统中导入了很多"联系人"(Contact)记录,它们没有具体所属的"客户"(Account)记录.那么我们就要 ...

  3. 人脸识别Android SDK集成

    目前我们的应用内使用了ArcFace 的人脸检测功能,这里就和大家分享一下我们的集成过程和一些使用心得~ 集成ArcFace FD 的集成过程非常简单在 ArcFace FD 的文档上有说明支持的系统 ...

  4. DVWA 黑客攻防演练(六)不安全的验证码 Insecure CAPTCHA

    之前在 CSRF 攻击 的那篇文章的最后,我觉得可以用验证码提高攻击的难度. 若有验证码的话,就比较难被攻击者利用 XSS 漏洞进行的 CSRF 攻击了,因为要识别验证码起码要调用api,跨域会被浏览 ...

  5. ICD

    International Classification of Diseases,ICD 国际疾病分类

  6. STP生成树协议

    STP主要作用 1.消除环路:通过阻断冗余链路来消除网络中可能存在的链路 2.链路备份:当活动那个路径发生故障时,激活备份链路,及时恢复网络连通性. 根桥选举 每个交换机启动STP后,都认为自己是根桥 ...

  7. Redis学习笔记(5)——Redis数据持久化

    出处http://www.cnblogs.com/xiaoxi/p/7065328.html 一.概述 Redis的强大性能很大程度上都是因为所有数据都是存储在内存中的,然而当Redis重启后,所有存 ...

  8. Zabbix 3.4.7针对一些主机设置期间维护

    场景说明: 由于公司有些主机设置了定时开机关机,每次开机关机得时候都会发邮件告警,每次都需要值班人员提醒,为了处理这种无效告警,可以在zabbix中设置维护 zabbix中的维护---维护期间:用来设 ...

  9. Android 动态设置TextView的drawableLeft等属性

    首先,我们在开发过程中,会经常使用到android:drawableLeft="@drawable/ic_launcher"这些类似的属性: 关于这些属性的意思,无非是在你的tex ...

  10. 返回数组中指定的一列,将键值作为元素键名array_column

    array_column() 函数 从记录集中取出 last_name 列: <?php // 表示由数据库返回的可能记录集的数组 $a = array( array( 'id' => 5 ...