class Solution {
public:
int stoneGameV(vector<int>& stoneValue) {
}
};
1563. 石子游戏 V
几块石子 排成一行 ,每块石子都有一个关联值,关联值为整数,由数组 stoneValue
给出。
游戏中的每一轮:Alice 会将这行石子分成两个 非空行(即,左侧行和右侧行);Bob 负责计算每一行的值,即此行中所有石子的值的总和。Bob 会丢弃值最大的行,Alice 的得分为剩下那行的值(每轮累加)。如果两行的值相等,Bob 让 Alice 决定丢弃哪一行。下一轮从剩下的那一行开始。
只 剩下一块石子 时,游戏结束。Alice 的分数最初为 0
。
返回 Alice 能够获得的最大分数 。
示例 1:
输入:stoneValue = [6,2,3,4,5,5] 输出:18 解释:在第一轮中,Alice 将行划分为 [6,2,3],[4,5,5] 。左行的值是 11 ,右行的值是 14 。Bob 丢弃了右行,Alice 的分数现在是 11 。 在第二轮中,Alice 将行分成 [6],[2,3] 。这一次 Bob 扔掉了左行,Alice 的分数变成了 16(11 + 5)。 最后一轮 Alice 只能将行分成 [2],[3] 。Bob 扔掉右行,Alice 的分数现在是 18(16 + 2)。游戏结束,因为这行只剩下一块石头了。
示例 2:
输入:stoneValue = [7,7,7,7,7,7,7] 输出:28
示例 3:
输入:stoneValue = [4] 输出:0
提示:
1 <= stoneValue.length <= 500
1 <= stoneValue[i] <= 10^6
原站题解
java 解法, 执行用时: 366 ms, 内存消耗: 43.3 MB, 提交时间: 2023-08-18 15:21:52
class Solution { public int stoneGameV(int[] s) {// 把参数名改了一下...., 原参数名有点长.... int len = s.length; int[] sum = new int[len];// 前缀和数组 int[][] dp = new int[len][len]; sum[0] = s[0]; for (int i = 1; i < len; i++) { sum[i] += sum[i - 1] + s[i]; } for (int i = len - 2; i >= 0; i--) {// 从倒数第二行开始dp for (int j = i + 1; j < len; j++) { dp[i][j] = Integer.MIN_VALUE; for (int k = i + 1; k <= j; k++) { int r = sum[j] - sum[k - 1]; int l = sum[k - 1] - sum[i] + s[i]; if (l > r){ dp[i][j] = Math.max(dp[i][j], r + dp[k][j]); }else if (l < r){ dp[i][j] = Math.max(dp[i][j], l + dp[i][k - 1]); }else { dp[i][j] = Math.max(dp[i][j], r + Math.max(dp[k][j], dp[i][k - 1])); } } } } return dp[0][len - 1]; } }
python3 解法, 执行用时: 1160 ms, 内存消耗: 29.7 MB, 提交时间: 2023-08-18 15:20:24
class Solution: def stoneGameV(self, stoneValue: List[int]) -> int: n = len(stoneValue) f = [[0] * n for _ in range(n)] maxl = [[0] * n for _ in range(n)] maxr = [[0] * n for _ in range(n)] for left in range(n - 1, -1, -1): maxl[left][left] = maxr[left][left] = stoneValue[left] total = stoneValue[left] suml = 0 i = left - 1 for right in range(left + 1, n): total += stoneValue[right] while i + 1 < right and (suml + stoneValue[i + 1]) * 2 <= total: suml += stoneValue[i + 1] i += 1 if left <= i: f[left][right] = max(f[left][right], maxl[left][i]) if i + 1 < right: f[left][right] = max(f[left][right], maxr[i + 2][right]) if suml * 2 == total: f[left][right] = max(f[left][right], maxr[i + 1][right]) maxl[left][right] = max(maxl[left][right - 1], total + f[left][right]) maxr[left][right] = max(maxr[left + 1][right], total + f[left][right]) return f[0][n - 1]