BZOJ3622 已经没有什么好害怕的了 【dp + 二项式反演】
题目链接
题解
既已开题
那就已经没有什么好害怕的了
由题目中奇怪的条件我们可以特判掉\(n - k\)为奇数时答案为\(0\)
否则我们要求的就是糖果大于药片恰好有\(\frac{n - k}{2} + k\)个的方案数,我们记为\(K\)
思路1##
直接求恰好不好求,想到二项式反演:
如果有
\]
那么有
\]
如果我们令\(a_k\)为恰好有\(K\)对糖果大于药片的方案数
从式子来看,对于每一种\(a_i\)的情况,\(b_k\)都从中选出任意\(k\)个
那么\(b_k\)就是任选\(k\)个满足条件,剩余随便选的方案数
问题就转化为了求出\(b_k\),显然可以\(dp\)
我们先将糖果与药片分别升序排序
我们设\(f[i][j]\)表示前\(i\)个糖果选出\(j\)个和药片匹配,都大于药片的方案数
我们记\(pos_i\)为第\(i\)个糖果比药片大的最大的位置,那么如果第\(i\)个糖果参与匹配,就有\(pos_i\)种选择
由于糖果是升序排序,所以剩余\(j - 1\)个糖果选择范围一定不比\(i\)大,所以\(i\)实际剩余\(pos_i - (j - 1)\)种选择
那么有
\]
求出\(f[n][i]\),则\(b_k = f[n][k] * (n - k)!\)
然后再用二项式反演即可求出\(a_k\)
思路2##
如果你不知道二项式反演,其实有一个更直观的方法
同样是求出\(f[n][i]\)后求出\(b_i\)
我们令\(ans[k]\)表示恰好\(k\)个糖果大于药片的方案数
显然
\]
当\(k < n\)时,\(b_k\)多算了恰好为\(k + 1\)、\(k + 2\)、\(k + 3......\)的方案数,减去即可
\]
这样也是对的
PS##
由思路2其实我们可以看到二项式反演实际上就是一个容斥的结果
思路一
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<map>
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long long int
using namespace std;
const int maxn = 2005,maxm = 100005,INF = 1000000000,P = 1000000009;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
LL f[maxn][maxn],C[maxn][maxn],fac[maxn];
int a[maxn],b[maxn];
int n,K;
void init(){
fac[0] = 1;
for (int i = 1; i <= n; i++) fac[i] = fac[i - 1] * i % P;
for (int i = 0; i <= n; i++){
C[i][0] = C[i][i] = 1;
for (int j = 1; j <= (i >> 1); j++)
C[i][j] = C[i][i - j] = (C[i - 1][j - 1] + C[i - 1][j]) % P;
}
sort(a + 1,a + 1 + n);
sort(b + 1,b + 1 + n);
}
int main(){
n = read(); K = read();
if ((n - K) & 1) {puts("0"); return 0;}
K = ((n - K) >> 1) + K;
for (int i = 1; i <= n; i++) a[i] = read();
for (int i = 1; i <= n; i++) b[i] = read();
init();
f[0][0] = 1;
for (int i = 1; i <= n; i++){
f[i][0] = f[i - 1][0];
int pos = n;
while (pos && b[pos] >= a[i]) pos--;
for (int j = 1; j <= i; j++)
f[i][j] = (f[i - 1][j] + f[i - 1][j - 1] * max(pos - j + 1,0) % P) % P;
}
LL ans = 0,t;
for (int i = K; i <= n; i++){
t = ((i - K) & 1) ? -1 : 1;
ans = ((ans + t * C[i][K] % P * (f[n][i] * fac[n - i] % P) % P) % P + P) % P;
}
printf("%lld\n",ans);
return 0;
}
思路2
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<map>
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long long int
using namespace std;
const int maxn = 2005,maxm = 100005,INF = 1000000000,P = 1000000009;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
LL f[maxn][maxn],C[maxn][maxn],fac[maxn],ans[maxn];
int a[maxn],b[maxn];
int n,K;
void init(){
fac[0] = 1;
for (int i = 1; i <= n; i++) fac[i] = fac[i - 1] * i % P;
for (int i = 0; i <= n; i++){
C[i][0] = C[i][i] = 1;
for (int j = 1; j <= (i >> 1); j++)
C[i][j] = C[i][i - j] = (C[i - 1][j - 1] + C[i - 1][j]) % P;
}
sort(a + 1,a + 1 + n);
sort(b + 1,b + 1 + n);
}
int main(){
n = read(); K = read();
if ((n - K) & 1) {puts("0"); return 0;}
K = ((n - K) >> 1) + K;
for (int i = 1; i <= n; i++) a[i] = read();
for (int i = 1; i <= n; i++) b[i] = read();
init();
f[0][0] = 1;
for (int i = 1; i <= n; i++){
f[i][0] = f[i - 1][0];
int pos = n;
while (pos && b[pos] >= a[i]) pos--;
for (int j = 1; j <= i; j++)
f[i][j] = (f[i - 1][j] + f[i - 1][j - 1] * max(pos - j + 1,0) % P) % P;
}
ans[n] = f[n][n];
for (int i = n - 1; i >= K; i--){
ans[i] = f[n][i] * fac[n - i] % P;
for (int j = i + 1; j <= n; j++)
ans[i] = (ans[i] - C[j][i] * ans[j] % P) % P;
ans[i] = (ans[i] + P) % P;
}
printf("%lld\n",ans[K]);
return 0;
}
BZOJ3622 已经没有什么好害怕的了 【dp + 二项式反演】的更多相关文章
- P4859 已经没有什么好害怕的了(dp+二项式反演)
P4859 已经没有什么好害怕的了 啥是二项式反演(转) 如果你看不太懂二项式反演(比如我) 那么只需要记住:对于某两个$g(i),f(i)$ ---------------------------- ...
- bzoj3622已经没有什么好害怕的了 dp+组合+容斥(?)
3622: 已经没有什么好害怕的了 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1033 Solved: 480[Submit][Status][ ...
- [BZOJ3622] 已经没有什么好害怕的了(dp+容斥)
Description: 有两个数组a和b,两两配对,求 \(a_i>b_i\) 的配对比 \(b_i>a_i\) 的配对多 \(k\) 个的方案数 \(k\le n\le 2000\ ...
- [bzoj3622]已经没有什么好害怕的了_动态规划_容斥原理
bzoj-3622 已经没有什么好害怕的了 题目大意: 数据范围:$1\le n \le 2000$ , $0\le k\le n$. 想法: 首先,不难求出药片比糖果小的组数. 紧接着,我开始的想法 ...
- bzoj3622已经没有什么好害怕的了
bzoj3622已经没有什么好害怕的了 题意: 给n个数Ai,n个数Bi,将Ai中的数与Bi中的数配对,求配对Ai比Bi大的比Bi比Ai大的恰好有k组的方案数.n,k≤2000 题解: 蒟蒻太弱了只能 ...
- BZOJ 3622 : 已经没有什么好害怕的了(dp + 广义容斥原理)
今天没听懂 h10 的讲课 但已经没有什么好害怕的了 题意 给你两个序列 \(a,b\) 每个序列共 \(n\) 个数 , 数之间两两不同 问 \(a\) 与 \(b\) 之间有多少配对方案 使得 \ ...
- luoguP4859 已经没有什么好害怕的了(二项式反演)
luoguP4859 已经没有什么好害怕的了(二项式反演) 祭奠天国的bzoj. luogu 题解时间 先特判 $ n - k $ 为奇数无解. 为了方便下记 $ m = ( n + k ) / 2 ...
- [BZOJ3622]已经没有什么好害怕的了(容斥DP)
给定两个数组a[n]与b[n](数全不相等),两两配对,求“a比b大”的数对比“b比a大”的数对个数多k的配对方案数. 据说做了这题就没什么题好害怕的了,但感觉实际上这是一个套路题,只是很难想到. 首 ...
- BZOJ3622 已经没有什么好害怕的了 动态规划 容斥原理 组合数学
原文链接https://www.cnblogs.com/zhouzhendong/p/9276479.html 题目传送门 - BZOJ3622 题意 给定两个序列 $a,b$ ,各包含 $n$ 个数 ...
随机推荐
- HDU.3177Crixalis's Equipment(贪心)
题目来源:3177 题目分析:一只蝎子要搬动一堆装备到一个容量为V的洞里面,每个装备有两个属性,一个是固有体积A,放置之后洞的剩余空间就会减少A,一个是移动体积B,只有当体积B小于等于当前洞的剩余体积 ...
- DCMTK读取DICOM文件头信息的三种方法
Howto: Load File Meta-Header Here's an example that shows how to load the File Meta Information Head ...
- MappingException:class com.zsn.crm.Model.user not found whie looking for property user id
之前好好地运行 什么东西都没动过 再次运行突然报异常*****MappingException:class com.zsn.crm.Model.user not found whie looking ...
- vim指令,快捷键汇总
Vim 命令.操作.快捷键全集 命令历史 以:和/开头的命令都有历史纪录,可以首先键入:或/然后按上下箭头来选择某个历史命令. 启动vim 在命令行窗口中输入以下命令即可 vim 直接启动vim vi ...
- datatable中reload和load的区别
ajax.reload()用于datatable表某个数据的变化而重新加载 ajax.url(url).load() 用于切换url时,datatable重新获取数据,加载.
- openldap完整版本搭建记录
文档信息 目 的:搭建一套完整的OpenLDAP系统,实现账号的统一管理. 1:OpenLDAP服务端的搭建 ...
- pytorch中如何使用预训练词向量
不涉及具体代码,只是记录一下自己的疑惑. 我们知道对于在pytorch中,我们通过构建一个词向量矩阵对象.这个时候对象矩阵是随机初始化的,然后我们的输入是单词的数值表达,也就是一些索引.那么我们会根据 ...
- 【机器学习算法基础+实战系列】SVM
概述 支持向量机是一种二分类模型,间隔最大使它有别于感知机.支持向量机学习方法由简至繁的模型:线性可分支持向量机(linear support vector machine in linearly s ...
- (洛谷)P2709 小B的询问
题目描述 小B有一个序列,包含N个1~K之间的整数.他一共有M个询问,每个询问给定一个区间[L..R],求Sigma(c(i)^2)的值,其中i的值从1到K,其中c(i)表示数字i在[L..R]中的重 ...
- JSP---JSTL核心标签库的使用
JSTL 核心标签库标签共有13个,功能上分为4类: 1.表达式控制标签:out.set.remove.catch 2.流程控制标签:if.choose.when.otherwise 3.循环标签:f ...