P3205 [HNOI2010]合唱队(区间dp+方案数)

P3205 [HNOI2010]合唱队 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

这道题 大区间包括小区间,每加一个人都会让区间更大;

考虑区间DP:

对于区间 [ i ~ j ] ,这段区间最新进的人只有两种可能 i 或 j

所以我们定义:

f[i][j]最后一个加的人是 i ,组成区间[ i ~ j ]的方案数 

g[i][j]最后一个加的人是 j ,组成区间[ i ~ j ]的方案数  

对于 f [i][j] 的转移:

f[ i ][ j ]是由区间[ i+1~j ]转移过来的,且a[i] 是要小于上一个加入的大小

对于区间[ i+1 ~ j ]同理也是有两种可能 (i+1)是这段区间最后一个加入的 或 j是这段区间最后一个加入的

如果 (i+1)是区间[i+1,j]最后一个加入的,且为了让 i 是最后一个加入的,要满足(a[i+1]>a[i])则有转移方程 f[i][j]+=f[i+1][j];

如果 (j)是区间[i+1,j]最后一个加入的,且为了让 i 是最后一个加入的,要满足(a[j]>a[i])则有转移方程 f[i][j]+=g[i+1][j];

 

对于 g [i][j] 的转移:

g[ i ][ j ]是由区间[ i~j-1 ]转移过来的,且a[j]是大于上一个加入的大小

对于区间[ i ~ j-1 ]同理也是有两种可能 i 是这段区间最后一个加入的 或 j-1 是这段区间最后一个加入的

如果 i 是区间[i,j-1]最后一个加入的,且为了让 j 是最后一个加入的,要满足(a[j]>a[i])则有转移方程 g[i][j]+=f[i][j-1];

如果 j-1 是区间[i,j-1]最后一个加入的,且为了让 j 是最后一个加入的,要满足(a[j]>a[j-1])则有转移方程 g[i][j]+=g[i][j-1];

 

对于初始化状态:

for(int i=1;i<=n;i++) f[i][i]=1;
 初始化状态不能有g[i][i]=1,(n=1的情况就能推翻,可根据常理来想)
 
Code:
#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define mp make_pair
#define pb push_back   
#define popb pop_back  
#define fi first
#define se second
const int N=1e3+10;
const int MOD=19650827;
//const int inf=0x3f3f3f3f;     
//const ll INF=0x3ffffffffffff;
int T,n,a[N],f[N][N],g[N][N];
inline int read()
{
    int x=0,f=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){x=(x<<1)+(x<<3)+(ch^48);ch=getchar();}
    return x*f;
}
int main()
{
//    freopen("","r",stdin);
//    freopen("","w",stdout);
    n=read();
    for(int i=1;i<=n;i++) a[i]=read(); 
    for(int i=1;i<=n;i++) f[i][i]=1;
    for(int len=2;len<=n;len++)
        for(int i=1;i+len-1<=n;i++)
            {
                int j=i+len-1;
                f[i][j]=(f[i+1][j]*(a[i+1]>a[i])+g[i+1][j]*(a[j]>a[i]))%MOD;
                g[i][j]=(f[i][j-1]*(a[j]>a[i])+g[i][j-1]*(a[j]>a[j-1]))%MOD;
            }
    printf("%d",(f[1][n]+g[1][n])%MOD);
    return 0;
}

 

 

 

posted @ 2023-03-01 22:51  QAQ啥也不会  阅读(35)  评论(0编辑  收藏  举报