NOIP模拟赛 czy的后宫4

czy的后宫4  

【问题描述】  

czy有很多妹子,妹子虽然数量很多,但是质量不容乐观,她们的美丽值全部为负数(喜闻乐见)。

czy每天都要带N个妹子到机房,她们都有一个独一无二的美丽值,美丽值为-1到-N之间的整数。他想要把这些妹子排成一个波动序列,这样相对“漂亮”(美丽值的绝对值较小)的妹子可以与她旁边的两个美丽值的绝对值较大的妹子形成鲜明的对比,整个序列相对将更加“美观”(不再那么无法直视)。

一个序列是波动序列仅当序列中的每个数比周围的两个数都大或都小(如果有的话)。

现在czy希望知道,长度为N的波动序列有多少种。两种序列A和B不同当且仅当存在一个i,使得Ai≠Bi。由于这个数目可能很大,你只对它除以P的余数感兴趣。  

【输入格式】  

输入文件czy.in仅含一行,两个正整数N, P。  

【输出格式】  

输出文件czy.out仅含一行,一个非负整数,表示你所求的答案对P取余之后的结果。  

【样例输入输出】 

czy.in   

4 7

czy.out    

3  

说明:共有10种可能的序列,它们是:  1324 1423 2143 2314 2413  3142 3241 3412 4132 4231

(忽略负号)  

【数据规模和约定】  

对于20%的数据,满足N≤10;  

对于40%的数据,满足N≤18; 

对于70%的数据,满足N≤550;  

对于100%的数据,满足3≤N≤4200,P≤10^9。 

 

F[i][j]=F[i][j-1]+F[i-1]+[i-j]

F[i][j]:i表示最大数n为i时,前j个数的方案数

最后总方案数乘以2

 1 #include<iostream>
 2 using namespace std;
 3 
 4 int n,p;
 5 int f[4201][4201];
 6 
 7 int read()
 8 {
 9     int x=0,f=1;char ch=getchar();
10     while(ch<'0'||ch>'9'){if(ch=='-')f=-f;ch=getchar();}
11     while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
12     return x*f;
13 }
14 
15 int main()
16 {
17     n=read();p=read();
18     f[1][1]=1;
19     for(int i=2;i<=n;i++)
20         for(int j=1;j<=n;j++)
21             f[i][j]=(f[i][j-1]+f[i-1][i-j])%p;
22     printf("%d\n",n==1?1:f[n][n]*2%p);
23     return 0;
24 }

 

posted @ 2016-11-09 20:15  InWILL  阅读(302)  评论(0编辑  收藏  举报