Python每日一练(20230509) 石子游戏 IV\V\VI

简介: Python每日一练(20230509) 石子游戏 IV\V\VI

1. 石子游戏 Stone Game IV


Alice 和 Bob 两个人轮流玩一个游戏,Alice 先手。


一开始,有 n 个石子堆在一起。每个人轮流操作,正在操作的玩家可以从石子堆里拿走 任意 非零 平方数 个石子。


如果石子堆里没有石子了,则无法操作的玩家输掉游戏。


给你正整数 n ,且已知两个人都采取最优策略。如果 Alice 会赢得比赛,那么返回 True ,否则返回 False 。


示例 1:

输入:n = 1

输出:true

解释:Alice 拿走 1 个石子并赢得胜利,因为 Bob 无法进行任何操作。


示例 2:

输入:n = 2

输出:false

解释:Alice 只能拿走 1 个石子,然后 Bob 拿走最后一个石子并赢得胜利(2 -> 1 -> 0)。


示例 3:

输入:n = 4

输出:true

解释:n 已经是一个平方数,Alice 可以一次全拿掉 4 个石子并赢得胜利(4 -> 0)。


示例 4:

输入:n = 7

输出:false

解释:当 Bob 采取最优策略时,Alice 无法赢得比赛。


如果 Alice 一开始拿走 4 个石子, Bob 会拿走 1 个石子,然后 Alice 只能拿走 1 个石子,Bob 拿走最后一个石子并赢得胜利(7 -> 3 -> 2 -> 1 -> 0)。


如果 Alice 一开始拿走 1 个石子, Bob 会拿走 4 个石子,然后 Alice 只能拿走 1 个石子,Bob 拿走最后一个石子并赢得胜利(7 -> 6 -> 2 -> 1 -> 0)。


示例 5:

输入:n = 17

输出:false

解释:如果 Bob 采取最优策略,Alice 无法赢得胜利。


提示:

   1 <= n <= 10^5

代码:

class Solution:
    def stoneGameIV(self, n: int) -> bool:
        dp = [False] * (n + 1)
        for i in range(1, n + 1):
            for j in range(1, int(i**0.5) + 1):
                if not dp[i - j*j]:
                    dp[i] = True
                    break
        return dp[n]
#%%
s = Solution()
print(s.stoneGameIV(1))
print(s.stoneGameIV(2))
print(s.stoneGameIV(4))
print(s.stoneGameIV(7))
print(s.stoneGameIV(17))

输出:

True

False

True

False

False


2. 石子游戏 Stone Game 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


代码1: 动态规划

from typing import List
class Solution:
    def stoneGameV(self, stoneValue: List[int]) -> int:
        n = len(stoneValue)
        presum = [0] * (n + 1)
        for i in range(n):
            presum[i + 1] = presum[i] + stoneValue[i]
        dp = [[0] * n for _ in range(n)]
        for i in range(n):
            dp[i][i] = 0
        for i in range(n - 2, -1, -1):
            for j in range(i + 1, n):
                for k in range(i, j):
                    leftsum = presum[k + 1] - presum[i]
                    rightsum = presum[j + 1] - presum[k + 1]
                    if leftsum < rightsum:
                        dp[i][j] = max(dp[i][j], dp[i][k] + leftsum)
                    elif leftsum > rightsum:
                        dp[i][j] = max(dp[i][j], dp[k + 1][j] + rightsum)
                    else:
                        dp[i][j] = max(dp[i][j], dp[i][k] + leftsum, dp[k + 1][j] + rightsum)
        return dp[0][n - 1]
#%%
s = Solution()
stoneValue = [6,2,3,4,5,5]
print(s.stoneGameV(stoneValue))
stoneValue = [7,7,7,7,7,7,7]
print(s.stoneGameV(stoneValue))


代码2:记忆化搜索

from typing import List
class Solution:
    def stoneGameV(self, stoneValue: List[int]) -> int:
        n = len(stoneValue)
        presum = [0] * (n + 1)
        for i in range(n):
            presum[i + 1] = presum[i] + stoneValue[i]
        memo = [[-1] * n for _ in range(n)]
        def dfs(i, j):
            if memo[i][j] != -1:
                return memo[i][j]
            if i == j:
                memo[i][j] = 0
                return 0
            res = 0
            for k in range(i, j):
                leftsum = presum[k + 1] - presum[i]
                rightsum = presum[j + 1] - presum[k + 1]
                if leftsum < rightsum:
                    res = max(res, dfs(i, k) + leftsum)
                elif leftsum > rightsum:
                    res = max(res, dfs(k + 1, j) + rightsum)
                else:
                    res = max(res, dfs(i, k) + leftsum, dfs(k + 1, j) + rightsum)
            memo[i][j] = res
            return res
        return dfs(0, n - 1)
#%%
s = Solution()
stoneValue = [6,2,3,4,5,5]
print(s.stoneGameV(stoneValue))
stoneValue = [7,7,7,7,7,7,7]
print(s.stoneGameV(stoneValue))

输出:

18

28


3. 石子游戏 Stone Game VI


Alice 和 Bob 轮流玩一个游戏,Alice 先手。


一堆石子里总共有 n 个石子,轮到某个玩家时,他可以 移出 一个石子并得到这个石子的价值。Alice 和 Bob 对石子价值有 不一样的的评判标准 。双方都知道对方的评判标准。


给你两个长度为 n 的整数数组 aliceValues 和 bobValues 。aliceValues[i] 和 bobValues[i] 分别表示 Alice 和 Bob 认为第 i 个石子的价值。


所有石子都被取完后,得分较高的人为胜者。如果两个玩家得分相同,那么为平局。两位玩家都会采用 最优策略 进行游戏。


请你推断游戏的结果,用如下的方式表示:


   如果 Alice 赢,返回 1 。

   如果 Bob 赢,返回 -1 。

   如果游戏平局,返回 0 。


示例 1:

输入:aliceValues = [1,3], bobValues = [2,1]

输出:1


解释:

如果 Alice 拿石子 1 (下标从 0开始),那么 Alice 可以得到 3 分。

Bob 只能选择石子 0 ,得到 2 分。

Alice 获胜。


示例 2:

输入:aliceValues = [1,2], bobValues = [3,1]

输出:0

解释:

Alice 拿石子 0 , Bob 拿石子 1 ,他们得分都为 1 分。

打平。


示例 3:

输入:aliceValues = [2,4,3], bobValues = [1,6,7]

输出:-1

解释:

不管 Alice 怎么操作,Bob 都可以得到比 Alice 更高的得分。

比方说,Alice 拿石子 1 ,Bob 拿石子 2 , Alice 拿石子 0 ,Alice 会得到 6 分而 Bob 得分为 7 分。

Bob 会获胜。


提示:

   n == aliceValues.length == bobValues.length

   1 <= n <= 105

   1 <= aliceValues[i], bobValues[i] <= 100


代码1: 动态规划


from typing import List
class Solution:
    def stoneGameVI(self, aliceValues: List[int], bobValues: List[int]) -> int:
        n = len(aliceValues)
        total_values = [(aliceValues[i] + bobValues[i], i) for i in range(n)]
        total_values.sort(reverse=True)
        alice_dp, bob_dp = [0] * (n + 1), [0] * (n + 1)
        for i in range(n):
            if i % 2 == 0:
                alice_dp[i // 2 + 1] = alice_dp[i // 2] + aliceValues[total_values[i][1]]
            else:
                bob_dp[i // 2 + 1] = bob_dp[i // 2] + bobValues[total_values[i][1]]
        for i in range(n // 2):
            alice_dp[i + 1] = max(alice_dp[i], alice_dp[i + 1])
            bob_dp[i + 1] = max(bob_dp[i], bob_dp[i + 1])
        diff = alice_dp[n // 2] - bob_dp[n // 2]
        if diff > 0:
            return 1
        elif diff < 0:
            return -1
        else:
            return 0
#%%
s = Solution()
aliceValues = [1,3]
bobValues = [2,1]
print(s.stoneGameVI(aliceValues, bobValues))
aliceValues = [1,2]
bobValues = [3,1]
print(s.stoneGameVI(aliceValues, bobValues))
aliceValues = [2,4,3]
bobValues = [1,6,7]
print(s.stoneGameVI(aliceValues, bobValues))


代码2: 贪心算法

from typing import List
class Solution:
    def stoneGameVI(self, aliceValues: List[int], bobValues: List[int]) -> int:
        n = len(aliceValues)
        stones = [(aliceValues[i] + bobValues[i], i) for i in range(n)]
        stones.sort(reverse=True)
        a_score, b_score = 0, 0
        for i in range(n):
            if i % 2 == 0:
                a_score += aliceValues[stones[i][1]]
            else:
                b_score += bobValues[stones[i][1]]
        if a_score > b_score:
            return 1
        elif a_score < b_score:
            return -1
        else:
            return 0
#%%
s = Solution()
aliceValues = [1,3]
bobValues = [2,1]
print(s.stoneGameVI(aliceValues, bobValues))
aliceValues = [1,2]
bobValues = [3,1]
print(s.stoneGameVI(aliceValues, bobValues))
aliceValues = [2,4,3]
bobValues = [1,6,7]
print(s.stoneGameVI(aliceValues, bobValues))


输出:

1

0

-1

目录
相关文章
|
3月前
|
小程序 PHP 图形学
热门小游戏源码(Python+PHP)下载-微信小程序游戏源码Unity发实战指南​
本文详解如何结合Python、PHP与Unity开发并部署小游戏至微信小程序。涵盖技术选型、Pygame实战、PHP后端对接、Unity转换适配及性能优化,提供从原型到发布的完整指南,助力开发者快速上手并发布游戏。
|
5月前
|
存储 算法 区块链
从零实现Python扫雷游戏:完整开发指南与深度解析
扫雷作为Windows经典游戏,承载了许多人的童年回忆。本文将详细介绍如何使用Python和Tkinter库从零开始构建一个功能完整的扫雷游戏,涵盖游戏设计、算法实现和界面开发的全过程。
398 0
|
6月前
|
人工智能 搜索推荐 数据可视化
用 Python 制作简单小游戏教程:手把手教你开发猜数字游戏
本教程详细讲解了用Python实现经典猜数字游戏的完整流程,涵盖从基础规则到高级功能的全方位开发。内容包括游戏逻辑设计、输入验证与错误处理、猜测次数统计、难度选择、彩色输出等核心功能,并提供完整代码示例。同时,介绍了开发环境搭建及调试方法,帮助初学者快速上手。最后还提出了图形界面、网络对战、成就系统等扩展方向,鼓励读者自主创新,打造个性化游戏版本。适合Python入门者实践与进阶学习。
688 1
|
6月前
|
存储 算法 数据可视化
用Python开发猜数字游戏:从零开始的手把手教程
猜数字游戏是编程入门经典项目,涵盖变量、循环、条件判断等核心概念。玩家通过输入猜测电脑生成的随机数,程序给出提示直至猜中。项目从基础实现到功能扩展,逐步提升难度,适合各阶段Python学习者。
416 0
|
3月前
|
数据采集 机器学习/深度学习 人工智能
Python:现代编程的首选语言
Python:现代编程的首选语言
290 102
|
3月前
|
数据采集 机器学习/深度学习 算法框架/工具
Python:现代编程的瑞士军刀
Python:现代编程的瑞士军刀
314 104
|
3月前
|
人工智能 自然语言处理 算法框架/工具
Python:现代编程的首选语言
Python:现代编程的首选语言
261 103
|
3月前
|
机器学习/深度学习 人工智能 数据挖掘
Python:现代编程的首选语言
Python:现代编程的首选语言
193 82
|
2月前
|
Python
Python编程:运算符详解
本文全面详解Python各类运算符,涵盖算术、比较、逻辑、赋值、位、身份、成员运算符及优先级规则,结合实例代码与运行结果,助你深入掌握Python运算符的使用方法与应用场景。
179 3

推荐镜像

更多