博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
洛谷 [P2051] 中国象棋
阅读量:5241 次
发布时间:2019-06-14

本文共 1465 字,大约阅读时间需要 4 分钟。

DP

orz__stdcall

首先要想出来,每行最多只能放两个棋子,这是显然的

于是决策就是一行一行地处理

30分的做法就是裸的枚举,暴搜,枚举这一行放哪里,放几个

然后想到了压位dp,按3进制表示当前棋盘的状态,即某一列没有棋子,或者有一个,两个棋子,能过50分

接着可以发现,棋子的顺序是无所谓的,并不需要准确知道当前棋盘的状态

于是有了100分做法:dp[i][j][k]表示放了前i行,有j列是有1个棋子,有k列有两个棋子

然后枚举所有的转移即可

#include 
#include
#include
#include
#include
using namespace std;const int MOD = 9999973;long long n, m, dp[105][105][105];long long C(long long num) { return num * (num - 1) / 2;}int main() { cin >> n >> m; dp[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++) { dp[i][j][k] += dp[i - 1][j][k] % MOD; if(j > 0) (dp[i][j][k] += dp[i - 1][j - 1][k] * (m - j - k + 1)) %= MOD; if(j < m && k > 0) (dp[i][j][k] += dp[i - 1][j + 1][k - 1] * (j + 1)) %= MOD; if(j > 1) (dp[i][j][k] += dp[i - 1][j - 2][k] * C(m - j - k + 2)) %= MOD; if(k > 0) (dp[i][j][k] += dp[i - 1][j][k - 1] * (m - j - k + 1) * j) %= MOD; if(j < m - 1 && k > 1) (dp[i][j][k] += dp[i - 1][j + 2][k - 2] * C(j + 2)) %= MOD; } } } long long ans = 0ll; for(int i = 0; i <= m; i++) { for(int j = 0; j + i <= m; j++) { (ans += dp[n][i][j]) %= MOD; } } cout << ans << endl; return 0;}

转载于:https://www.cnblogs.com/Mr-WolframsMgcBox/p/9073685.html

你可能感兴趣的文章
图论-次短路求法
查看>>
What's New for Visual C# 6.0
查看>>
ExtJs学习笔记之ComboBox组件
查看>>
关于收费软件
查看>>
getopt_long
查看>>
TensorFlow MNIST CNN 代码
查看>>
javascript之Style物
查看>>
JSON跨域解决方案收集
查看>>
SSH框架整合总结
查看>>
图的深度优先遍历
查看>>
C# 之 提高WebService性能大数据量网络传输处理
查看>>
md5sum命令详解
查看>>
[bzoj1004] [HNOI2008] Cards
查看>>
应该是实例化对象的没有对属性赋值时,自动赋值为null,但不是空指针对象引用...
查看>>
原生HttpClient详细使用示例
查看>>
几道面试题
查看>>
Factory Design Pattern
查看>>
python中贪婪与非贪婪
查看>>
guava API整理
查看>>
无锁编程笔记
查看>>