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

    你可能感兴趣的文章
    Oracle11g静默安装dbca,netca报错处理--直接跟换操作系统
    查看>>
    oracle12安装软件后安装数据库,然后需要自己配置监听
    查看>>
    Oracle——08PL/SQL简介,基本程序结构和语句
    查看>>
    Oracle——distinct的用法
    查看>>
    Oracle、MySQL、SQL Server架构大对比
    查看>>
    oracle下的OVER(PARTITION BY)函数介绍
    查看>>
    Oracle中DATE数据相减问题
    查看>>
    Oracle中merge into的使用
    查看>>
    oracle中sql查询上月、本月、上周、本周、昨天、今天的数据!
    查看>>
    oracle中sql的case语句运用--根据不同条件去排序!
    查看>>
    Oracle中Transate函数的使用
    查看>>
    oracle中关于日期问题的汇总!
    查看>>
    Oracle中常用的语句
    查看>>
    oracle中新建用户和赋予权限
    查看>>
    Oracle中的NVL,NVL2,NULLIF以及COALESCE函数使用
    查看>>
    Oracle中的rownum 和rowid的用法和区别
    查看>>
    oracle中的大小写、字符、dual、数字、处理、日期、函数、显/隐式、时间、条件表达式case、decode、to_date、to_char、sysdate
    查看>>
    Oracle修改字段类型
    查看>>
    oracle典型安装失败,安装oracle 10失败
    查看>>
    Oracle分析函数之LEAD和LAG
    查看>>