列表

详情


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

 

提示:

原站题解

去查看

上次编辑到这里,代码来自缓存 点击恢复默认模板
class Solution { public: int stoneGameV(vector<int>& stoneValue) { } };

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]

上一题