dp[i]表示一定包含第I个点的好的子序列个数,那么最终答案就是求dp[0] + dp[1] + .... + dp[n-1]
最终的子序列被分成了很多块,因此很明显我们枚举第一块,第一块和剩下的再去组合,然后我们为了保证没有重复,我们需要保证第一块不同,然而第一块的大小是固定的,因此我们可以选择枚举第一块最后一个数,这样第一块就肯定不会相同了,也可以计算
const ll P = 998244353; ll dp[maxn]; int N = 1000; ll comb[maxn][maxn]; int main(){ for(int i=0;i<=N;i++) comb[i][0]=comb[i][i]=1; for(int i=2;i<=N;i++) for(int j=1;j<N;j++) comb[i][j]=(comb[i-1][j]+comb[i-1][j-1])%P; int n; cin >> n; vector<int> a(n); for(int i = 0 ; i < n ; i++) { cin >> a[i]; } dp[n]=1; ll ans=0; for(int i = n - 1 ; i >= 0 ; i--){ if(a[i] <= 0 ) continue; ll sum = 0; for(int j = n ; j >= i + a[i] + 1 ; j--){ sum = (sum + dp[j]) % P; dp[i] = (dp[i] + comb[j - i - 2][a[i] -1] % P * sum % P) % P; } //cout<<dp[i]<<endl; //cout<<ans<<endl; ans= (ans+dp[i]) %P; } cout<<ans<<endl; return 0; }
原文地址:https://www.cnblogs.com/033000-/p/12322668.html
时间: 2024-10-11 16:48:37