P7 UVA11481 Arrange the Numbers

UVA11481 Arrange the Numbers

组合数问题。

做法貌似很多,显然在前 m 个数中选 k 个,即 C(m,k),然后后面有 mk 个数需要保证不放在自己的位置上,所以后面整体是一个禁位问题,貌似可以用棋盘多项式去推禁位公式,但是暂时不会。不过还有另外一种思路,就是对于后面的 nm 个 “自由数”,可以考虑枚举有多少数放在原位,这样显然可以包含所有情况,那么对于剩下的就是经典的错排问题了。复杂度为 O(n2+Tn),OK。

关于错排递推公式 D(n)=(n1)[D(n1)+D(n2)] 的证明:考虑第 n 个数可以放在 1n1 这些位置上,比如放在第 i 位,那么对于 i ,如果把它放在 n 这个位置上,那么就变成了 D(n2) 这个位置;如果把它放在其它位置上,发现实际上其本质就变成了 D(n1) 这个问题。

当然错排公式也可以由容斥定理推出 D(n)=n!i=0n1i!(1)i

const int N=1005,mod=1e9+7;

ll fac[N],d[N],c[N][N];

signed main(){
  IOS
  fac[0]=1;
  for(int i=1;i<=1000;++i) fac[i]=fac[i-1]*i%mod;
  c[0][0]=1;
  for(int i=1;i<=1000;++i){
    c[i][0]=1;
    for(int j=1;j<=i;++j)
      c[i][j]=(c[i-1][j]+c[i-1][j-1])%mod;
  }//combinatorial number

  d[0]=1,d[1]=0;
  for(int i=2;i<=1000;++i)
    d[i]=(d[i-1]+d[i-2])%mod*(i-1)%mod;

  int T;
  cin>>T; int n,m,k;
  for(int kase=1;kase<=T;++kase){
    cin>>n>>m>>k; ll ans=0;
    for(int i=0;i<=n-m;++i) ans=(ans+d[m-k+i]*c[n-m][i]%mod)%mod;
    cout<<"Case "<<kase<<": "<<ans*c[m][k]%mod<<'\n';
  }
  
  return 0;
}

· EOF

posted @   ComplexityMFC  阅读(18)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示