Description

给n个人安排座位,先给每个人一个1~n的编号,设第i个人的编号为ai(不同人的编号可以相同),接着从第一个人开始,大家依次入座,第i个人来了以后尝试坐到ai,如果ai被占据了,就尝试ai+1,ai+1也被占据了的话就尝试ai+2,……,如果一直尝试到第n个都不行,该安排方案就不合法。然而有m个人的编号已经确定(他们或许贿赂了你的上司...),你只能安排剩下的人的编号,求有多少种合法的安排方案。由于答案可能很大,只需输出其除以M后的余数即可。

Input

第一行一个整数T,表示数据组数

对于每组数据,第一行有三个整数,分别表示n、m、M

若m不为0,则接下来一行有m对整数,p1、q1,p2、q2 ,…, pm、qm,其中第i对整数pi、qi表示第pi个人的编号必须为qi

Output

对于每组数据输出一行,若是有解则输出YES,后跟一个整数表示方案数mod M,注意,YES和数之间只有一个空格,否则输出NO

Sample Input

2

4 3 10

1 2 2 1 3 1

10 3 8882

7 9 2 9 5 10

Sample Output

YES 4

NO

HINT

100%的数据满足:1≤T≤10,1≤n≤300,0≤m≤n,2≤M≤109,1≤pi、qi≤n   且保证pi互不相同。

正解:DP+组合数学

解题报告:

  考虑一个方案如果合法,那么对于任何一个编号i,都可以保证编号小于等于i的人数大于等于i,可以yy一下,肯定是对的,如果不满足的话直接输出NO就可以了。

  然后我们统计一下那些固定的编号,另外的非固定的编号我们可以都视为0,因为实际上这些人是没有任何区别的而且可以随意编号。

  我们接着统计一下前缀和,只要发现不合法了就直接break掉。

  然后就是DP部分啦,f[i][j]表示有j个人编号小于等于i的方案数,显然i<=j,那么我们可以考虑如何从i-1转移到i。因为是从i-1转移过来,那么显然我们需要知道编号确定为i的人究竟有多少个。所以我们可以枚举这一位,也就是i,编号恰好为i的人数k。那么这个人数k显然最少为num[i],也就是编号必须为i的人数,另外我们需要知道小于等于i的人有j个,所以同样需要枚举。j的范围是i到sum[i](最多为sum[i],相当于是能选编号小于等于i的最大人数)。所以上一位一共选了j-k个人(这一位新选了k个,一共是j个人),所以就可以从f[i-1][j-k]转移过来。那么这一位一共要选出k-num[i]个人编号为i,因为num[i]是必须要编号为i的人数,所以剩下的才是可供自由支配的。然后总人数呢?就是sum[i]-(j-k)也就是说这一位最多选sum[i],上一位选了(j-k),可知这一位最多选sum[i]-num[i]-j+k个人,也就是可以自由支配的总数。所以只需要把上一位的方案数f[i-1][j-k]乘上一个组合数就可以得到这次的f[i][j]。所以我们要预处理一下组合数。记得取模!

  代码如下:

 //It is made by jump~
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <ctime>
#include <vector>
#include <queue>
#include <map>
#include <set>
using namespace std;
typedef long long LL;
const int inf = (<<);
const int MAXN = ;
#define RG register
int n,m,MOD;
int sum[MAXN],num[MAXN];
LL C[MAXN][MAXN];
LL f[MAXN][MAXN];//f[i][j]表示有j个人编号小于等于i的方案数,显然i<=j inline int getint()
{
RG int w=,q=; RG char c=getchar();
while((c<'' || c>'') && c!='-') c=getchar(); if(c=='-') q=,c=getchar();
while (c>='' && c<='') w=w*+c-'', c=getchar(); return q ? -w : w;
} inline void work(){
RG int T=getint(); RG int x;
while(T--) {
memset(sum,,sizeof(sum)); memset(num,,sizeof(num)); memset(C,,sizeof(C)); memset(f,,sizeof(f));
n=getint();m=getint();MOD=getint(); C[][]=;
for(RG int i=;i<=n;i++) { C[i][]=;for(RG int j=;j<=i;j++) C[i][j]=C[i-][j-]+C[i-][j],C[i][j]%=MOD; }//递推组合数
for(RG int i=;i<=m;i++) x=getint(),x=getint(),num[x]++;
RG bool flag=true; sum[]=n-m;//默认为编号为0
for(int i=;i<=n;i++) { sum[i]=sum[i-]+num[i]; if(sum[i]<i) { flag=false; break; } }
if(!flag) { printf("NO\n"); continue;}
f[][]=;//初值
for(RG int i=;i<=n;i++)
for(RG int j=i;j<=sum[i];j++) {//这一个编号有多少个人
for(RG int k=num[i];k<=j-i+;k++)//这一个编号新选k个人,那么上一个编号选了j-k个,这一位最多j-i+1个,即上一位选了i-1个
f[i][j]+=f[i-][j-k]*C[sum[i]-num[i]-j+k][k-num[i]],f[i][j]%=MOD;//一共有sum[i]-num[i]-(j-k)个人可以选,这一位必须要选的有num[i]个,然后自由选的从中选就有k-num[i]个
}
printf("YES %lld\n",f[n][n]);
}
} int main()
{
work();
return ;
}

BZOJ2302 [HAOI2011]Problem c的更多相关文章

  1. BZOJ2302 [HAOI2011]Problem c 【dp】

    题目 给n个人安排座位,先给每个人一个1~n的编号,设第i个人的编号为ai(不同人的编号可以相同),接着从第一个人开始,大家依次入座,第i个人来了以后尝试坐到ai,如果ai被占据了,就尝试ai+1,a ...

  2. 【BZOJ2302】[HAOI2011]Problem C(动态规划)

    [BZOJ2302][HAOI2011]Problem C(动态规划) 题面 BZOJ 洛谷 题解 首先如果\(m=0\)即没有特殊限制的话,那么就和这道题目基本上是一样的. 然而这题也有属于这题的性 ...

  3. BZOJ2301: [HAOI2011]Problem b[莫比乌斯反演 容斥原理]【学习笔记】

    2301: [HAOI2011]Problem b Time Limit: 50 Sec  Memory Limit: 256 MBSubmit: 4032  Solved: 1817[Submit] ...

  4. bzoj 2301: [HAOI2011]Problem b

    2301: [HAOI2011]Problem b Time Limit: 50 Sec Memory Limit: 256 MB Submit: 3757 Solved: 1671 [Submit] ...

  5. HAOI2011 problem b

    2301: [HAOI2011]Problem b Time Limit: 50 Sec  Memory Limit: 256 MBSubmit: 1047  Solved: 434[Submit][ ...

  6. BZOJ 2298: [HAOI2011]problem a 动态规划

    2298: [HAOI2011]problem a Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnli ...

  7. BZOJ 2301: [HAOI2011]Problem b 莫比乌斯反演

    2301: [HAOI2011]Problem b Time Limit: 50 Sec  Memory Limit: 256 MBSubmit: 1007  Solved: 415[Submit][ ...

  8. 2301: [HAOI2011]Problem b

    2301: [HAOI2011]Problem b Time Limit: 50 Sec  Memory Limit: 256 MBSubmit: 4164  Solved: 1888[Submit] ...

  9. BZOJ 2302: [HAOI2011]Problem c( dp )

    dp(i, j)表示从i~N中为j个人选定的方案数, 状态转移就考虑选多少人为i编号, 然后从i+1的方案数算过来就可以了. 时间复杂度O(TN^2) ------------------------ ...

随机推荐

  1. uGUI练习 开篇

    一.准备阶段 1.Unity 4.6 Beta b18或更高版本(注:目前泄露版的Unity5.0Beta 对UI的支持并没有4.6 Beta那么好) 2.了解 Unity 2D Sprite的基础知 ...

  2. js知识体系的梳理一

    今天简单的总结了js的一些东西,梳理下整个体系,每一次的总结都会有不同的收获:js总结一一.[获取元素]: 1.通过ID: var oBtn=document.getElementById('btn1 ...

  3. Android开发EditText属性

    Android开发EditText属性 EditText继承关系:View-->TextView-->EditText EditText的属性很多,这里介绍几个:android:hint= ...

  4. work_queue 函数调用栈

    init_workqueues --->  create_worker --> kthread_create_on_node

  5. 012医疗项目-模块一:统一异常处理器的设计思路及其实现(涉及到了Springmvc的异常处理流程)

    我们上一篇文章是建立了一个自定义的异常类,来代替了原始的Exception类.在Serice层抛出异常,然后要在Action层捕获这个异常,这样的话在每个Action中都要有try{}catch{}代 ...

  6. 1017. A除以B (20)

    本题要求计算A/B,其中A是不超过1000位的正整数,B是1位正整数.你需要输出商数Q和余数R,使得A = B * Q + R成立. 输入格式: 输入在1行中依次给出A和B,中间以1空格分隔. 输出格 ...

  7. Linux下Qt的安装与配置

    参考资料:http://www.cnblogs.com/emouse/archive/2013/01/28/2880142.html Linux 下编译.安装.配置 QT 下载qt 这里用的是4.7. ...

  8. HttpClient, 使用C#操作Web

    我们知道, .Net类库里提供了HttpWebRequest等类,方便我们编程与Web服务器进行交互. 但是实际使用中我们经常会遇到以下需求,基础类里没有直接提供相应的功能(WebClient类包含这 ...

  9. GPS坐标换算为百度坐标

    最近在做一个关于手机定位的小应用,需求是这样的,用户通过手机(Wp8)进行二维码扫描操作并且记录用户的当前位置,在PC上可以查看用户所在地图的位置,做法就是在用户扫描条码时,通过手机GPS获取当前在地 ...

  10. matlab中fopen 和 fprintf函数总结

    matlab中fopen函数在指定文件打开的实例如下: *1)"fopen"打开文件,赋予文件代号. 语法1:FID= FOPEN(filename,permission) 用指定 ...