木題大戰Vol.0 DP搬運工1
阿新 • • 發佈:2020-08-17
木題大戰Vol.0 DP搬運工1
題目描述
給你 \(n\),\(K\),求有多少個 \(1\) 到 \(n\) 的排列,滿足相鄰兩個數的 \(max\) 的和不超過 \(K\) 。
分析
預設型\(DP\)
我們定義狀態 \(f[i][j][k]\) 表示填完 $1∼i \(、有 j\) $個位置可以填數、貢獻總和為 \(k\) 的方案數
我們假定從小到大填數,對於一個數 \(x\) ,如果我們在之後的操作中在這一個數的兩邊都填入了新的數,那麼當前的數一定不會貢獻價值
如果只在一邊填入新的數,那麼當前的數貢獻的價值為 \(x\)
如果兩邊都不填入數,那麼貢獻的價值為 \(2x\)
如果新填入的點在數列兩邊的話,轉移方程為
\[f[i][j+1][k]=(f[i][j+1][k]+f[i-1][j][k]*2LL) \]
\[f[i][j][k+i]=(f[i][j][k+i]+f[i-1][j][k]*2LL) \]
如果新填入的數在中間的話,轉移方程為
\[f[i][j+1][k]=(f[i][j+1][k]+f[i-1][j][k]*j) \]
\[f[i][j][k+i]=(f[i][j][k+i]+f[i-1][j][k]*j*2LL) \]
\[f[i][j-1][k+i+i]=(f[i][j-1][k+i+i]+f[i-1][j][k]*j) \]
程式碼
#include<bits/stdc++.h> using namespace std; const int maxn=55; const long long mod=998244353; long long f[3][maxn][maxn*maxn]; int n,m; 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("D.in","r",stdin); freopen("D.out","w",stdout); n=read(),m=read(); f[1][0][0]=1; int now=1; for(int i=2;i<=n;i++){ now^=1; for(int j=0;j<maxn;j++){ for(int k=0;k<maxn*maxn;k++){ f[now][j][k]=0; } } int maxj=min(i,n-i)+1,maxk=min(m,i*i); for(int j=0;j<=maxj;j++){ for(int k=0;k<=maxk;k++){ if(f[now^1][j][k]==0) continue; f[now][j+1][k]=(f[now][j+1][k]+f[now^1][j][k]*2LL)%mod; f[now][j][k+i]=(f[now][j][k+i]+f[now^1][j][k]*2LL)%mod; if(j==0) continue; f[now][j+1][k]=(f[now][j+1][k]+f[now^1][j][k]*j)%mod; f[now][j][k+i]=(f[now][j][k+i]+f[now^1][j][k]*j*2LL)%mod; f[now][j-1][k+i+i]=(f[now][j-1][k+i+i]+f[now^1][j][k]*j)%mod; } } } long long ans=0; for(int i=0;i<=m;i++){ ans=(ans+f[now][0][i])%mod; } printf("%lld\n",ans); return 0; }