博客
关于我
HDU - 4597 Play Game (博弈 + 区间dp)
阅读量:284 次
发布时间:2019-03-01

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

Alice和Bob在玩一个拿卡片的游戏,有两个堆,每个堆有N张卡片。每张卡片都有一个分数。他们轮流从任意一堆的顶部或底部拿走一张卡片,拿走的卡片分数会加到自己的总分里。目标是找出Alice能拿到的最大可能分数,因为他先开始。

思路

这个问题可以通过动态规划和记忆化搜索来解决。我们需要记录每个可能的状态,也就是每个堆剩下的卡片情况,以及当前是Alice还是Bob的回合。使用四维数组dp[i][j][p][q]表示在当前状态下,玩家能够获得的最大分数。

  • 状态定义dp[i][j][p][q]表示第一个堆剩余ij张卡片,第二个堆剩余pq张卡片时,当前玩家能够获得的最大分数。
  • 递归关系:当前玩家可以选择四种操作:拿走第一个堆的第一个或最后一个卡片,或者拿走第二个堆的第一个或最后一个卡片。每种选择都会导致对方玩家在新的状态下选择最优策略。
  • 记忆化:避免重复计算相同的状态,使用记忆化技术存储已经计算过的结果。
  • 解决代码

    #include 
    using namespace std;typedef long long ll;const int inf = 0x3f3f3f3f;int N = 25;int a[N], b[N];int dp[N][N][N][N];int n;ll dfs(int i, int j, int p, int q) { if (i > j && p > q) return 0; if (i <= j && p <= q) { int res = 0; if (i <= j) { res = max(res, dfs(i+1, j, p, q) + a[i]); res = max(res, dfs(i, j-1, p, q) + a[j]); } if (p <= q) { res = max(res, dfs(i, j, p+1, q) + b[p]); res = max(res, dfs(i, j, p, q-1) + b[q]); } dp[i][j][p][q] = res; return res; } if (i <= j) { ll take_a = a[i]; ll take_b = a[j]; ll option1 = dfs(i+1, j, p, q) + take_a; ll option2 = dfs(i, j-1, p, q) + take_b; dp[i][j][p][q] = max(option1, option2); } if (p <= q) { ll take_a = b[p]; ll take_b = b[q]; ll option3 = dfs(i, j, p+1, q) + take_a; ll option4 = dfs(i, j, p, q-1) + take_b; dp[i][j][p][q] = max(dp[i][j][p][q], option3, option4); } return dp[i][j][p][q];}int main() { int t; scanf("%d", &t); for(int _ = 0; _ < t; _++) { n = 0; int sum = 0; for(int i = 0; i < n; i++) { sum += a[i]; } for(int i = 0; i < n; i++) { sum += b[i]; } for(int i = 0; i < n; i++) { a[i] = 0; } for(int i = 0; i < n; i++) { b[i] = 0; } n = 0; scanf("%d", &n); for(int i = 0; i < n; i++) { int x; scanf("%d", &x); a[i] = x; } for(int i = 0; i < n; i++) { int x; scanf("%d", &x); b[i] = x; } int dp[N][N][N][N]; memset(dp, 0, sizeof(dp)); ll total = 0; for(int i = 0; i < n; i++) { total += a[i]; } for(int i = 0; i < n; i++) { total += b[i]; } ll res = dfs(0, n-1, 0, n-1); cout << res << endl; } return 0;}

    代码解释

  • 输入处理:读取测试用例数T,然后处理每个测试用例,读取N、数组a和数组b。
  • 动态规划数组初始化:使用四维数组dp存储每个状态的最大分数。
  • 递归函数dfs:计算当前状态下的最大分数,考虑四种操作,递归调用,并使用记忆化存储结果。
  • 输出结果:对每个测试用例,调用递归函数并输出结果。
  • 通过这种方法,我们可以高效地解决问题,确保Alice能获得的最大分数。

    转载地址:http://kcio.baihongyu.com/

    你可能感兴趣的文章
    OSPF技术连载2:OSPF工作原理、建立邻接关系、路由计算
    查看>>
    OSPF技术连载5:OSPF 基本配置,含思科、华为、Junifer三厂商配置
    查看>>
    OSPF技术连载6:OSPF 多区域,近7000字,非常详细!
    查看>>
    OSPF技术连载7:什么是OSPF带宽?OSPF带宽参考值多少?
    查看>>
    OSPF技术连载8:OSPF认证:明文认证、MD5认证和SHA-HMAC验证
    查看>>
    OSPF故障排除技巧
    查看>>
    spring配置文件中<context:property-placeholder />的使用
    查看>>
    OSPF有哪些优势?解决了RIP的什么问题?
    查看>>
    OSPF理论
    查看>>
    OSPF的七种类型LSA
    查看>>
    OSPF的安全性考虑:全面解析与最佳实践
    查看>>
    OSPF知识点大全,网络工程师快速收藏!
    查看>>
    ospf综合实验2 2012/9/8
    查看>>
    OSPF规划两大模型:双塔奇兵、犬牙交错
    查看>>
    OSPF认证
    查看>>
    OSPF设计原则,命令以H3C为例
    查看>>
    ospf路由 华3_动态路由OSPF基本原理及配置,一分钟了解下
    查看>>
    OSPF路由协议配置
    查看>>
    OSPRay 开源项目教程
    查看>>
    VC++实现应用程序对插件的支持
    查看>>