[bzoj3622]已经没有什么好害怕的了_动态规划_容斥原理
bzoj-3622 已经没有什么好害怕的了
题目大意:

数据范围:$1\le n \le 2000$ , $0\le k\le n$。
想法:
首先,不难求出药片比糖果小的组数。
紧接着,我开始的想法是
$f_{(i,j)}$表示前$i$个糖果中,满足糖果比药片大的组数是$j$的方案数。
进而发现需要将两个数组排序。
到这里一切都很正常,但是我们发现了一个问题:就是我在转移的时候,分两种情况讨论。第一种是当前糖果配对的药片比自己大,第二种是比自己小。
这样的话我需要乘上两个组合数。
但是我们仔细思考一下:如果这样转移的话,排序的意义(是的前面的区间不影响后面的区间)就失效了,我们发现这鬼东西是个有后效性的转移。
然后啊....通常我们遇到有后效性的$dp$怎么办呢?
这个后效性根本没有办法制约。
看了$cqzhangyu$的题解恍然大悟。
哦原来还可以容斥掉。
我们修改一下上面那个状态
$f_{(i,j)}$表示前$i$个糖果中,满足糖果比药片大的组数至少为$j$,且只考虑“糖果比药片大的糖果”的摆放情况的方案数。
这样的话我们就,暴力转移一下就行了。
就还是像上面一样分类讨论,但是如果是讨论比自己大的情况就直接加。
但是统计的时候需要乘上组合数,因为需要把比药片小的糖果的情况乘一下就好了嗷。
代码:
#include <bits/stdc++.h>
#define N 2010
using namespace std;
#define mod 1000000009
int a[N],b[N];
typedef long long ll;
ll f[N][N],fac[N],c[N][N];
ll ans=0;
char *p1,*p2,buf[100000];
#define nc() (p1==p2&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++)
int rd() {int x=0,f=1; char c=nc(); while(c<48) {if(c=='-') f=-1; c=nc();} while(c>47) x=(((x<<2)+x)<<1)+(c^48),c=nc(); return x*f;}
int main()
{
int n=rd(),m=rd();
if((n+m) % 2) puts("0"),exit(0);
m=(n+m)/2;
fac[0]=1; for(int i=1;i<=n;i++) fac[i]=fac[i-1] * i % mod;
for(int i=1;i<=n;i++) a[i]=rd();
for(int i=1;i<=n;i++) b[i]=rd();
for(int i=0;i<=n;i++)
{
c[i][0]=1;
for(int j=1;j<=i;j++) c[i][j] = (c[i-1][j-1] + c[i-1][j]) % mod;
}
sort(a+1,a+n+1); sort(b+1,b+n+1);
f[0][0]=1;
for(int i=1;i<=n;i++)
{
int k;
for(k=1;k<=n && b[k] < a[i];k++);
k--; for(int j=1;j<=i;j++) f[i][j]=(f[i-1][j] + f[i-1][j-1] * max(k-j+1,0)) % mod;
f[i][0]=f[i-1][0];
}
ll tmp=1;
for(int i=m;i<=n;i++) f[n][i]=(f[n][i]*fac[n-i])%mod,ans=(ans + tmp*f[n][i]*c[i][m]%mod + mod) % mod,tmp*=(-1);
cout << (ans + mod) % mod << endl ;
return 0;
}
小结:做一个后效性的$dp$,另一种办法就是采用容斥原理。
[bzoj3622]已经没有什么好害怕的了_动态规划_容斥原理的更多相关文章
- BZOJ3622 已经没有什么好害怕的了(动态规划+容斥原理)
显然可以转化为一个阶梯状01矩阵每行每列取一个使权值和为k的方案数.直接做不可做,考虑设f[i][j]为前i行权值和至少为j,即在其中固定了j行选1的方案数.设第i行从1~a[i]列都是1且a[i]+ ...
- 【BZOJ3622】已经没有什么好害怕的了(动态规划,容斥)
[BZOJ3622]已经没有什么好害怕的了(动态规划,容斥) 题面 BZOJ 题解 很明显的,这类问题是要从至少变成恰好的过程,直接容斥即可. 首先我们要求的是(糖果>药片)=(药片>糖果 ...
- bzoj3622已经没有什么好害怕的了
bzoj3622已经没有什么好害怕的了 题意: 给n个数Ai,n个数Bi,将Ai中的数与Bi中的数配对,求配对Ai比Bi大的比Bi比Ai大的恰好有k组的方案数.n,k≤2000 题解: 蒟蒻太弱了只能 ...
- [BZOJ3622]已经没有什么好害怕的了(容斥DP)
给定两个数组a[n]与b[n](数全不相等),两两配对,求“a比b大”的数对比“b比a大”的数对个数多k的配对方案数. 据说做了这题就没什么题好害怕的了,但感觉实际上这是一个套路题,只是很难想到. 首 ...
- BZOJ3622 已经没有什么好害怕的了 【dp + 二项式反演】
题目链接 BZOJ3622 题解 既已开题 那就已经没有什么好害怕的了 由题目中奇怪的条件我们可以特判掉\(n - k\)为奇数时答案为\(0\) 否则我们要求的就是糖果大于药片恰好有\(\frac{ ...
- bzoj3622已经没有什么好害怕的了 dp+组合+容斥(?)
3622: 已经没有什么好害怕的了 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1033 Solved: 480[Submit][Status][ ...
- BZOJ3622 已经没有什么好害怕的了 动态规划 容斥原理 组合数学
原文链接https://www.cnblogs.com/zhouzhendong/p/9276479.html 题目传送门 - BZOJ3622 题意 给定两个序列 $a,b$ ,各包含 $n$ 个数 ...
- BZOJ3622 已经没有什么好害怕的了
Description Input Output Sample Input 4 2 5 35 15 45 40 20 10 30 Sample Output 4 HINT 输入的2*n个数字保证全不相 ...
- 【BZOJ3622】已经没什么好害怕的了 容斥原理+dp
Description Input Output Sample Input 4 2 5 35 15 45 40 20 10 30 Sample Output 4 HINT 输入的2*n个数字保证全不相 ...
随机推荐
- JDBC连接数据库详解
JDBC连接数据库 •创建一个以JDBC连接数据库的程序,包含7个步骤: 1.加载JDBC驱动程序: 在连接数据库之前,首先要加载想要连接的数据库的驱动到JVM(Java虚拟机),这通过java.la ...
- Python3 try-except、raise和assert解析
Python3 try-except.raise和assert解析 一.说明 关于异常捕获try-except:在学java的时候就被教育异常捕获也是java相对c的一大优点,几年下来多少也写了些代码 ...
- c++ 当输入的数据不符合数据类型时,清理输入流
if (!cin) { cin.clear(); while (cin.get() != '\n') continue; cout << "Bad input; input pr ...
- Bzoj 1055: [HAOI2008]玩具取名 (区间DP)
Bzoj 1055: [HAOI2008]玩具取名 (区间DP) 题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1055 区间动态规划和可 ...
- Re:从零开始的Linux之路(基础篇)
基于 Red Hat Enterprise Linux 7.5 或者 CentOS 7.4 Linux的命令一定遵循以下格式:command指令 [-options]选项 parameter1参数 ...
- js 字符串加密
加密: 1.获得要加密的字符串:var str=input.value; 2.转化: for(var i=0;i<str.length;i++){ str+=String.fromCharCod ...
- MySQL之单表查询、多表查询
一.单表查询: 单个表的查询方法及语法顺序需要通过实际例子来熟悉 先将表数据创建下: mysql> create database singe_t1; # 建个数据库singe_t1 Query ...
- nrf52810学习笔记——三
在开发nRF52系列的蓝牙方案的时候,会用到IDE.SDK.softdevice.nrfgoStudio等开发软件,这里做一个小小的总结. 首先,下载SDK,里面有适合keil4号iar7(iar8也 ...
- WIN10配置MAVEN
添加新的系统环境变量M2_HOME, 并设置其值为你安装的目录MAVEN_HOME=D:\Softwares\apache-maven-3.2.2. 更新系统PATH 变量, 添加;%M2_HOME% ...
- LeetCode答案(python)
1. 两数之和 给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 两个 整数,并返回他们的数组下标. 你可以假设每种输入只会对应一个答案.但是,你不能重复利用这 ...