原文链接https://www.cnblogs.com/zhouzhendong/p/9276479.html

题目传送门 - BZOJ3622

题意

  给定两个序列 $a,b$ ,各包含 $n$ 个数字。

  现在给 $a$ 中元素与 $b$ 中元素配对。问使得所有配对中 $a_?>b_?$ 的个数比 $a_?<b_?$ 的个数恰好多 $k$ 的方案总数。

  答案对 $10^9+9$ 取模,保证 $a$ 和 $b$ 中的所有数字互不相同。

  $n\leq 2000$

题解

  首先闭着眼睛排个序。

  然后,我们发现一个非常好的性质:如果要使得 $a_i$ 的配对 $b_?$ 满足 $a_i>b_?$ 那么,满足 $1\leq ?\leq c_i$ ,其中 $c_i$ 表示在 $1\leq j\leq n$ 中满足 $b_j<a_i$ 的最大 $j$ 值。

  我们先把问题转化一下,变成求恰好得到 $x$ 个 $a_?>b_?$ 的方案数。那么显然 $x=\cfrac{n+k}{2}$ 。如果 $x$ 不是整数,那么答案显然是 $0$ 。

  下面说的满足条件是指 $a_?>b_?$ 。

  于是我们考虑动态规划。

  记 $f_{i,j}$ 表示在 $a_1,a_2,...,a_i$ 中选择 $j$ 个,匹配比他小的 $b_i$ 的方案数。

  那么我们可以轻松确定转移方程:

$$f_{i,j}=f_{i-1,j}+f_{i-1,j-1}\times \max(c_i-j+1,0)$$

  我们假设 $f_i=f_{n,i}$ ,并设 $g_i$ 为配对所有的元素,恰好有 $i$ 个满足条件的方案数。

  于是下式成立:

$$f_i=\sum_{j=i}^n\binom{j}{i}g_j$$  

  回忆一下 $f$ 的定义,$f_i$ 表示在 $n$ 个里面选择 $i$ 个匹配 $i$ 个满足条件的方案数。

  考虑所有的最终情况。

  满足条件个数为 $j$ 的总共有 $g_j$ 种。每一个这样的方案中选择的 $j$ 个数中,任选 $i$ 个,所得到的方案,都会对 $f_i$ 有贡献。

  所以,$f_i=\sum_{j=0}^{n}\binom{j}{i}g_j=\sum_{j=i}^n\binom{j}{i}g_j$ 。

  于是我们来用 $f$ 表示 $g$ 。

$$\begin{eqnarray*}g_k&=&\sum_{i=k}^{n}g_i\binom{i}{k}\sum_{j=k}^{i}(-1)^{j-k}\binom{i-k}{j-k} \\&=&\sum_{i=k}^{n}g_i\sum_{j=k}^{i}(-1)^{j-k}\binom{i}{k}\binom{i-k}{j-k}\\&=&\sum_{i=k}^{n}g_i\sum_{j=k}^{i}(-1)^{j-k}\binom{i}{j}\binom{j}{k}\\&=&\sum_{i=k}^{n}(-1)^{i-k}\binom{i}{k}\sum_{j=i}^{n}g_j\binom{j}{i}\\&=&\sum_{i=k}^{n}f_i(-1)^{i-k}\binom{i}{k}\end{eqnarray*}$$

  然后按照这个公式算一算就可以了。

代码

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. const int N=2005,mod=1e9+9;
  4. int n,k,a[N],b[N],c[N];
  5. int C[N][N],Fac[N];
  6. int f[N][N];
  7. int main(){
  8. scanf("%d%d",&n,&k);
  9. for (int i=1;i<=n;i++)
  10. scanf("%d",&a[i]);
  11. for (int i=1;i<=n;i++)
  12. scanf("%d",&b[i]);
  13. sort(a+1,a+n+1),sort(b+1,b+n+1);
  14. for (int i=1;i<=n;i++)
  15. for (c[i]=c[i-1];c[i]<n&&b[c[i]+1]<a[i];c[i]++);
  16. for (int i=Fac[0]=1;i<=n;i++)
  17. Fac[i]=1LL*Fac[i-1]*i%mod;
  18. for (int i=0;i<=n;i++)
  19. C[i][i]=C[i][0]=1;
  20. for (int i=1;i<=n;i++)
  21. for (int j=1;j<i;j++)
  22. C[i][j]=(C[i-1][j]+C[i-1][j-1])%mod;
  23. for (int i=0;i<=n;i++)
  24. f[i][0]=1;
  25. for (int i=1;i<=n;i++)
  26. for (int j=1;j<=n;j++)
  27. f[i][j]=(1LL*f[i-1][j-1]*max(c[i]-j+1,0)+f[i-1][j])%mod;
  28. int ans=0,x=(n+k)/2;
  29. if (x*2!=n+k){
  30. puts("0");
  31. return 0;
  32. }
  33. for (int i=x;i<=n;i++)
  34. ans=(1LL*f[n][i]*C[i][x]%mod*Fac[n-i]*((i-x+1)%2*2-1)+ans)%mod;
  35. ans=(ans+mod)%mod;
  36. printf("%d",ans);
  37. return 0;
  38. }

  

BZOJ3622 已经没有什么好害怕的了 动态规划 容斥原理 组合数学的更多相关文章

  1. [bzoj3622]已经没有什么好害怕的了_动态规划_容斥原理

    bzoj-3622 已经没有什么好害怕的了 题目大意: 数据范围:$1\le n \le 2000$ , $0\le k\le n$. 想法: 首先,不难求出药片比糖果小的组数. 紧接着,我开始的想法 ...

  2. bzoj3622已经没有什么好害怕的了

    bzoj3622已经没有什么好害怕的了 题意: 给n个数Ai,n个数Bi,将Ai中的数与Bi中的数配对,求配对Ai比Bi大的比Bi比Ai大的恰好有k组的方案数.n,k≤2000 题解: 蒟蒻太弱了只能 ...

  3. [BZOJ3622]已经没有什么好害怕的了(容斥DP)

    给定两个数组a[n]与b[n](数全不相等),两两配对,求“a比b大”的数对比“b比a大”的数对个数多k的配对方案数. 据说做了这题就没什么题好害怕的了,但感觉实际上这是一个套路题,只是很难想到. 首 ...

  4. BZOJ3622 已经没有什么好害怕的了 【dp + 二项式反演】

    题目链接 BZOJ3622 题解 既已开题 那就已经没有什么好害怕的了 由题目中奇怪的条件我们可以特判掉\(n - k\)为奇数时答案为\(0\) 否则我们要求的就是糖果大于药片恰好有\(\frac{ ...

  5. bzoj3622已经没有什么好害怕的了 dp+组合+容斥(?)

    3622: 已经没有什么好害怕的了 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1033  Solved: 480[Submit][Status][ ...

  6. BZOJ3622 已经没有什么好害怕的了(动态规划+容斥原理)

    显然可以转化为一个阶梯状01矩阵每行每列取一个使权值和为k的方案数.直接做不可做,考虑设f[i][j]为前i行权值和至少为j,即在其中固定了j行选1的方案数.设第i行从1~a[i]列都是1且a[i]+ ...

  7. BZOJ3622 已经没有什么好害怕的了

    Description Input Output Sample Input 4 2 5 35 15 45 40 20 10 30 Sample Output 4 HINT 输入的2*n个数字保证全不相 ...

  8. 【BZOJ3622】已经没什么好害怕的了 容斥原理+dp

    Description Input Output Sample Input 4 2 5 35 15 45 40 20 10 30 Sample Output 4 HINT 输入的2*n个数字保证全不相 ...

  9. 洛谷 P4859 && BZOJ3622: 已经没有什么好害怕的了

    题目描述 给出 \(n\) 个数 \(a_i\)​ ,以及 \(n\) 个数 \(b_i\)​ ,要求两两配对使得 \(a>b\) 的对数减去 \(a<b\) 的对数等于 \(k\) . ...

随机推荐

  1. ASP.NET提供三种主要形式的缓存

    ASP.NET提供三种主要形式的缓存:页面级输出缓存.用户控件级输出缓存(或称为片段缓存)和缓存API.

  2. SQL Server循环

    1.普通循环 DECLARE @i int BEGIN WHERE Uid=@i --PRINT @i END 2.游标循环(没有事务) ---游标循环(没有事务) BEGIN DECLARE @a ...

  3. linux 过滤内存使用率并于计划任务结合来自动清理内存缓存

    过滤出内存使用率并进行判断 #!/bin/bash echo "###cleand free_cache script########" #memory usage mem_pus ...

  4. Python-计算机硬件基础

    预习计算机硬件基础1,CPU/内存.硬盘2,存储器 什么是编程语言什么是编程编程的目的运行程序的三大核心硬件重要:运行程序,硬件的工作过程 ------------------------------ ...

  5. JavaScript自定义对象

    一,概述 在Java语言中,我们可以定义自己的类,并根据这些类创建对象来使用,在Javascript中,我们也可以定义自己的类,例如定义User类.Hashtable类等等. 目前在Javascrip ...

  6. verilog-产生axis数据流

    首先这是产生aixs数据流的代码 `timescale 1ps/1ps `default_nettype none module axis_switch_0_example_master #( ) ( ...

  7. 高并发编程基础Synchronized与Volatile

    关键字Synchronized: 当使用Synchrnized (o) ,锁定 o 的时候,锁定的是 o 指向的堆内存中 new 出来的对象,而非 o 引用,当锁定 o 以后,一旦 o 指向了其他对象 ...

  8. 最短路径之Floyd-warshall算法

    哇咔咔,最喜欢这种算法了,算法简单,暴力解决: 可惜数据大点就解决不了问题了: 输入的数据是 第一行第一个数是city的数量n,第二个是路径数t, 接下来n行为a至b的距离 4 81 2 2 1 3 ...

  9. java----DOS命令

    dir /?   查看帮助 dir /s   查看当前的目录,以及子目录

  10. Imperial roads 非严格次小生成树

    cf测评姬比uva快了五倍... /* 不管这条边是不是在mst上,直接跑lca求出路径上的最大边w即可 ans=mst-w+dist(u,v) */ #include<bits/stdc++. ...