题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4529

郑厂长系列故事——N骑士问题

Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others)
Total Submission(s): 642    Accepted Submission(s): 315

Problem Description
  郑厂长不是正厂长
  也不是副厂长
  他根本就不是厂长
  还是那个腾讯公司的码农
  一个业余时间喜欢下棋的码农
  
  最近,郑厂长对八皇后问题很感兴趣,拿着国际象棋研究了好几天,终于研究透了。兴奋之余,坐在棋盘前的他又开始无聊了。无意间,他看见眼前的棋盘上只摆了八个皇后,感觉空荡荡的,恰好又发现身边还有几个骑士,于是,他想把这些骑士也摆到棋盘上去,当然棋盘上的一个位置只能放一个棋子。因为受八皇后问题的影响,他希望自己把这些骑士摆上去之后,也要满足每2个骑士之间不能相互攻击。
  现在郑厂长想知道共有多少种摆法,你能帮助他吗?

骑士的下法:
  每步棋先横走或直走一格,然后再往外斜走一格;或者先斜走一格,最后再往外横走或竖走一格(即走“日”字)。可以越子,没有"中国象棋"的"蹩马腿"限制。

 
Input
输入第一行为一个整数T(1<=T<=8),表示有T组测试数据;
每组数据首先是一个整数N(1<=n<=10),表示要摆N个骑士上去;
接下来是一个8*8的矩阵来描述一个棋盘,’.’表示这个位置是空的,’*’表示这个位置上已经放了皇后了;
数据中的初始棋盘保证是一个合法的八皇后摆法。
 
Output
对每组数据,请在一行内输出一个整数,表示合法的方案数。
 
Sample Input
2
1
*.......
....*...
.......*
.....*..
..*.....
......*.
.*......
...*....
2
*.......
....*...
.......*
.....*..
..*.....
......*.
.*......
...*....
 
Sample Output
56
1409
 
Source

题解:

1.与此题(POJ1185 炮兵阵地)类似。

2.设dp[i][j][s1][s2]为:到第i行,总共放了j个,第i-1行的状态为s1,第i行的状态为s2的情况数。

3.复杂度为8*10*(1<<8)*(1<<8)*(1<<8),理论上是会超时的,但因为有很多剪枝,所以可以把时间复杂度降下去。

代码如下:

  1. #include <iostream>
  2. #include <cstdio>
  3. #include <cstring>
  4. #include <algorithm>
  5. #include <vector>
  6. #include <cmath>
  7. #include <queue>
  8. #include <stack>
  9. #include <map>
  10. #include <string>
  11. #include <set>
  12. using namespace std;
  13. typedef long long LL;
  14. const double EPS = 1e-;
  15. const int INF = 2e9;
  16. const LL LNF = 9e18;
  17. const int MOD = 1e5;
  18. const int MAXN = <<;
  19.  
  20. int row[];
  21. int cnt[];
  22. int dp[][][MAXN][MAXN];
  23.  
  24. int main()
  25. {
  26. int T, n;
  27. char str[];
  28. scanf("%d", &T);
  29. while(T--)
  30. {
  31. scanf("%d", &n);
  32. for(int i = ; i<=; i++)
  33. {
  34. scanf("%s", str);
  35. row[i] = ;
  36. for(int j = ; j<; j++)
  37. row[i] += (str[j]=='*')*(<<j);
  38. }
  39.  
  40. for(int s = ; s<MAXN; s++)
  41. {
  42. cnt[s] = ;
  43. for(int j = ; j<; j++)
  44. if(s&(<<j))
  45. cnt[s]++;
  46. }
  47.  
  48. memset(dp, , sizeof(dp));
  49. dp[][][][] = ;
  50. for(int i = ; i<=; i++)
  51. {
  52. for(int j = ; j<=n; j++)
  53. {
  54. for(int s1 = ; s1<MAXN; s1++)
  55. {
  56. if(i>&&(s1&row[i-])) continue;
  57. if(cnt[s1]>n) continue;
  58. for(int s2 = ; s2<MAXN; s2++)
  59. {
  60. if(i>&&(s2&row[i-])) continue;
  61. if( (s1&(s2<<)) || (s1&(s2>>)) ) continue;
  62. if(cnt[s2]>n) continue;
  63. for(int s3 = ; s3<MAXN; s3++)
  64. {
  65. if(s3&row[i]) continue;
  66. if( (s1&(s3<<)) || (s1&(s3>>)) ) continue;
  67. if( (s2&(s3<<)) || (s2&(s3>>)) ) continue;
  68. if(j-cnt[s3]<) continue;
  69. dp[i][j][s2][s3] += dp[i-][j-cnt[s3]][s1][s2];
  70. }
  71. }
  72. }
  73. }
  74. }
  75.  
  76. LL ans = ;
  77. for(int s1 = ; s1<MAXN; s1++)
  78. for(int s2 = ; s2<MAXN; s2++)
  79. ans += dp[][n][s1][s2];
  80. printf("%lld\n", ans);
  81. }
  82. }

滚动数组:

  1. #include <iostream>
  2. #include <cstdio>
  3. #include <cstring>
  4. #include <algorithm>
  5. #include <vector>
  6. #include <cmath>
  7. #include <queue>
  8. #include <stack>
  9. #include <map>
  10. #include <string>
  11. #include <set>
  12. using namespace std;
  13. typedef long long LL;
  14. const double EPS = 1e-;
  15. const int INF = 2e9;
  16. const LL LNF = 9e18;
  17. const int MOD = 1e5;
  18. const int MAXN = <<;
  19.  
  20. int row[];
  21. int cnt[];
  22. int dp[][][MAXN][MAXN];
  23.  
  24. int main()
  25. {
  26. int T, n;
  27. char str[];
  28. scanf("%d", &T);
  29. while(T--)
  30. {
  31. scanf("%d", &n);
  32. for(int i = ; i<=; i++)
  33. {
  34. scanf("%s", str);
  35. row[i] = ;
  36. for(int j = ; j<; j++)
  37. row[i] += (str[j]=='*')*(<<j);
  38. }
  39.  
  40. for(int s = ; s<MAXN; s++)
  41. {
  42. cnt[s] = ;
  43. for(int j = ; j<; j++)
  44. if(s&(<<j))
  45. cnt[s]++;
  46. }
  47.  
  48. memset(dp[], , sizeof(dp[]));
  49. dp[][][][] = ;
  50. for(int i = ; i<=; i++)
  51. {
  52. memset(dp[i%], , sizeof(dp[i%]));
  53. for(int j = ; j<=n; j++)
  54. {
  55. for(int s1 = ; s1<MAXN; s1++)
  56. {
  57. if(i>&&(s1&row[i-])) continue;
  58. if(cnt[s1]>n) continue;
  59. for(int s2 = ; s2<MAXN; s2++)
  60. {
  61. if(i>&&(s2&row[i-])) continue;
  62. if( (s1&(s2<<)) || (s1&(s2>>)) ) continue;
  63. if(cnt[s2]>n) continue;
  64. for(int s3 = ; s3<MAXN; s3++)
  65. {
  66. if(s3&row[i]) continue;
  67. if( (s1&(s3<<)) || (s1&(s3>>)) ) continue;
  68. if( (s2&(s3<<)) || (s2&(s3>>)) ) continue;
  69. if(j-cnt[s3]<) continue;
  70. dp[i%][j][s2][s3] += dp[(i+)%][j-cnt[s3]][s1][s2];
  71. }
  72. }
  73. }
  74. }
  75. }
  76.  
  77. LL ans = ;
  78. for(int s1 = ; s1<MAXN; s1++)
  79. for(int s2 = ; s2<MAXN; s2++)
  80. ans += dp[][n][s1][s2];
  81. printf("%lld\n", ans);
  82. }
  83. }

HDU4529 郑厂长系列故事——N骑士问题 —— 状压DP的更多相关文章

  1. HDU 4529 郑厂长系列故事——N骑士问题 状压dp

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=4529 郑厂长系列故事--N骑士问题 Time Limit: 6000/3000 MS (Java/O ...

  2. hdu_4529_郑厂长系列故事——N骑士问题(状压DP)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=4529 题意:中文,不解释 题解:状压DP,dp[i][j][k][s]表示第i行当前用了j个骑士,i- ...

  3. HDU 4539 郑厂长系列故事——排兵布阵 状压dp

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=4539 郑厂长系列故事--排兵布阵 Time Limit: 10000/5000 MS (Java/O ...

  4. HDU 4539 郑厂长系列故事——排兵布阵 —— 状压DP

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4539 郑厂长系列故事——排兵布阵 Time Limit: 10000/5000 MS (Java/Ot ...

  5. HDU-4529 郑厂长系列故事——N骑士问题 状态压缩DP

    题意:给定一个合法的八皇后棋盘,现在给定1-10个骑士,问这些骑士不能够相互攻击的拜访方式有多少种. 分析:一开始想着搜索写,发现该题和八皇后不同,八皇后每一行只能够摆放一个棋子,因此搜索收敛的很快, ...

  6. hdu_4539_郑厂长系列故事——排兵布阵(状压DP|最大团)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=4539 题意:中文,不解释 题解:将每一行的状态压缩,然后进行DP,也可以用最大团做.这里我用的DP # ...

  7. HDU----(4519)郑厂长系列故事——体检

    郑厂长系列故事——体检 Time Limit: 500/200 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others)Total S ...

  8. HDU 4539郑厂长系列故事――排兵布阵(状压DP)

    HDU 4539  郑厂长系列故事――排兵布阵 基础的状压DP,首先记录先每一行可取的所哟状态(一行里互不冲突的大概160个状态), 直接套了一个4重循环居然没超时我就呵呵了 //#pragma co ...

  9. HDU 4539 郑厂长系列故事——排兵布阵

    http://acm.hdu.edu.cn/showproblem.php?pid=4539 郑厂长系列故事——排兵布阵 Time Limit: 10000/5000 MS (Java/Others) ...

随机推荐

  1. k8s学习(一)——kubectl与api-server之间的交互核心过程

    k8s的架构是用户使用kubectl工具对虚拟机资源进行各种各样的控制和定制. 而kubectl本身并不包含对其核心资源的访问与控制.而是通过http通信与api-server进行交互实现资源的管理. ...

  2. Esper epl语句实验

    基础代码见下,下文列举的实验都是在此程序基础上改动. all,snapshot,first String epl = "select * from appTable.win:time(5 s ...

  3. js 输入框增加删除操作

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  4. linux挂载硬盘

    最近有个生产机由于缓存问题,硬盘不太够用,所以就加载了一块硬盘,下面把挂载硬盘的过程记录下,供大家参考,有问题欢迎留言! fdisk –l ------------------------------ ...

  5. vue-cli配置文件详解

    转自: https://blog.csdn.net/Mr_YanYan/article/details/79233188

  6. apue学习笔记(第十章 信号)

    本章先对信号机制进行综述,并说明每种信号的一般用法. 信号概念 每个信号都有一个名字,这些名字都以3个字符SIG开头.在头文件<signal.h>中,信号名都被定义为正整形常量. 在某个信 ...

  7. mongodb:monogo和php整合

    1.到如下网址,下载php扩展包,找一个最新stable版的.

  8. IIS配置asp.net网站

    http://wenku.baidu.com/view/f8ce6c14767f5acfa0c7cd36.html

  9. vim 查找与替换

    一.vim 查找 1. 正向查找 / 与 反向查找 ? 2. 退出查找 <Esc> 3. 跳转到下一处匹配 n ,跳转到上一处匹配 N 4. /<CR> 正向跳转到相同模式的下 ...

  10. redis安装报错

    错误原因大概是这个: creating server tcp listening socket 127.0.0.1:6379: bind No error 解决方案如下按顺序输入如下命令就可以连接成功 ...