zoukankan      html  css  js  c++  java
  • 2018.10.23-dtoi-2004:象棋Chess(Chess)

    题目描述:

    在N行M列的棋盘上,放若干个炮可以是0个,使得没有任何一个炮可以攻击另一个炮。
    请问有多少种放置方法,中国像棋中炮的行走方式大家应该很清楚吧.

    输入:

    一行包含两个整数N,M,中间用空格分开.

    输出:

    输出所有的方案数,由于值比较大,输出其mod 9999973

    数据范围:

    100%的数据中N,M不超过100
    50%的数据中,N,M至少有一个数不超过8
    30%的数据中,N,M均不超过6

    算法标签:DP

    思路:

    一开始可能会想到状压DP,但是只有30分。考虑dp,用f[i][j][k]表示在前一行,有j行放了1个炮,k行放了两个炮。

    考虑转移式子:...太多了看代码把

    以下代码:

    #include<bits/stdc++.h>
    #define il inline
    #define LL long long
    #define _(d) while(d(isdigit(ch=getchar())))
    using namespace std;
    const int N=105,p=9999973;LL f[N][N][N],ans;int n,m;
    il int read(){int x,f=1;char ch;_(!)ch=='-'?f=-1:f;x=ch^48;_()x=(x<<1)+(x<<3)+(ch^48);return f*x;}
    il LL cal(int x){return ((LL)x*(LL)(x-1)/2ll)%p;}
    int main()
    {
        n=read();m=read();f[0][0][0]=1;
        for(int i=1;i<=n;i++)for(int j=0;j<=m;j++)for(int k=0;k+j<=m;k++){
            (f[i][j][k]+=f[i-1][j][k])%=p;
            if(k>=1)(f[i][j][k]+=f[i-1][j+1][k-1]*(LL)(j+1))%=p;
            if(j>=1)(f[i][j][k]+=f[i-1][j-1][k]*(LL)(m-k-j+1)%p)%=p;
            if(j>=2)(f[i][j][k]+=f[i-1][j-2][k]*cal(m-k-j+2)%p)%=p;
            if(k>=2)(f[i][j][k]+=f[i-1][j+2][k-2]*cal(j+2)%p)%=p;
            if(k>=1)(f[i][j][k]+=f[i-1][j][k-1]*(LL)j*(LL)(m-j-k+1)%p)%=p;
        }
        for(int i=0;i<=m;i++)for(int j=0;j+i<=m;j++)(ans+=f[n][i][j])%=p;
        printf("%lld
    ",ans);
      return 0;
    }
    View Code
  • 相关阅读:
    ucore 物理内存探测 lab2 附录A&B
    git diff 笔记
    操作系统Lab1 详解(boot|kern/debug)
    ucore os 前初始化
    第五讲 计算机体系结构 内存层次
    Django 的学习(1) 从建立到数据库操作
    ucore os 初始化
    操作系统 Lab1
    makefile 语法笔记 3
    flex布局注意事项
  • 原文地址:https://www.cnblogs.com/Jessie-/p/9843892.html
Copyright © 2011-2022 走看看