T - Permutation 题解(思维+dp)

题目链接

题目大意

给你一个数字n和长为n-1个字符串

字符串包含'<','>'

若s[i]='<' 则代表a[i]<a[i+1]

若s[i]='>' 则代表a[i]>a[i+1]

你要构造一个长度为n的全排列,使其满足条件

求方案数mod 1e9+7

题目思路

这个问题主要就是不知道如何保证自己构造的数组是一个全排列

又是一个神仙dp,

\(dp[i][j]\)代表[1,i]为全排列,且第i个元素为j

那么该怎么转移。

若为'<' \(dp[i][j]=\sum_{t=1}^{t=j-1} dp[i-1][t]\)

若为‘>’ \(dp[i][j]=\sum_{t=j}^{t=n} dp[i-1][t]\)

这个转移实属神奇,把j放到i-1个全排列后面,然后前i-1中大于等于j的元素全部加1即可

这样前i个元素又满足全排列,最后一个元素还是j,满足转移方程

然后前缀和优化即可,注意j<=i

代码

#include<set>
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<cstdio>
#include<vector>
#include<string>
#include<cstring>
#include<iostream>
#include<algorithm>
#define fi first
#define se second
#define debug printf(" I am here\n");
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
const ll INF=0x3f3f3f3f3f3f3f3f;
const int maxn=3e3+5,inf=0x3f3f3f3f,mod=1e9+7;
const double eps=1e-10;
int n,flag[maxn];
char s[maxn];
ll dp[maxn][maxn],pre[maxn][maxn];
int main(){
    scanf("%d %s",&n,s+1);
    for(int i=1;i<=n-1;i++){
        if(s[i]=='<'){
            flag[i]=1;
        }else{
            flag[i]=0;
        }
    }
    dp[1][1]=1;
    for(int i=1;i<=n;i++){
        pre[1][i]=1;
    }
    for(int i=2;i<=n;i++){
        for(int j=1;j<=n;j++){
            if(j>i){
                pre[i][j]=pre[i][j-1];
                continue;
            }
            if(flag[i-1]){
                dp[i][j]=pre[i-1][j-1]%mod;
            }else{
                dp[i][j]=((pre[i-1][n]-pre[i-1][j-1])%mod+mod)%mod;
            }
            pre[i][j]=(pre[i][j-1]+dp[i][j])%mod;
        }
    }
    printf("%lld\n",pre[n][n]);
    return 0;
}

posted @ 2020-11-25 21:34  hunxuewangzi  阅读(209)  评论(0编辑  收藏  举报