博客
关于我
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/

    你可能感兴趣的文章
    ntko文件存取错误_苹果推送 macOS 10.15.4:iCloud 云盘文件夹共享终于来了
    查看>>
    ntpdate 通过外网同步时间
    查看>>
    NTPD使用/etc/ntp.conf配置时钟同步详解
    查看>>
    NTP及Chrony时间同步服务设置
    查看>>
    NTP配置
    查看>>
    NUC1077 Humble Numbers【数学计算+打表】
    查看>>
    NuGet Gallery 开源项目快速入门指南
    查看>>
    NuGet(微软.NET开发平台的软件包管理工具)在VisualStudio中的安装的使用
    查看>>
    nuget.org 无法加载源 https://api.nuget.org/v3/index.json 的服务索引
    查看>>
    Nuget~管理自己的包包
    查看>>
    NuGet学习笔记001---了解使用NuGet给net快速获取引用
    查看>>
    nullnullHuge Pages
    查看>>
    NullPointerException Cannot invoke setSkipOutputConversion(boolean) because functionToInvoke is null
    查看>>
    null可以转换成任意非基本类型(int/short/long/float/boolean/byte/double/char以外)
    查看>>
    Numix Core 开源项目教程
    查看>>
    numpy
    查看>>
    NumPy 或 Pandas:将数组类型保持为整数,同时具有 NaN 值
    查看>>
    numpy 或 scipy 有哪些可能的计算可以返回 NaN?
    查看>>
    numpy 数组 dtype 在 Windows 10 64 位机器中默认为 int32
    查看>>
    numpy 数组与矩阵的乘法理解
    查看>>