C++前缀和算法的应用:石头游戏 VIII 原理源码测试用例

这篇具有很好参考价值的文章主要介绍了C++前缀和算法的应用:石头游戏 VIII 原理源码测试用例。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

本文涉及的基础知识点

C++算法:前缀和、前缀乘积、前缀异或的原理、源码及测试用例 包括课程视频

题目

Alice 和 Bob 玩一个游戏,两人轮流操作, Alice 先手 。
总共有 n 个石子排成一行。轮到某个玩家的回合时,如果石子的数目 大于 1 ,他将执行以下操作:
选择一个整数 x > 1 ,并且 移除 最左边的 x 个石子。
将 移除 的石子价值之 和 累加到该玩家的分数中。
将一个 新的石子 放在最左边,且新石子的值为被移除石子值之和。
当只剩下 一个 石子时,游戏结束。
Alice 和 Bob 的 分数之差 为 (Alice 的分数 - Bob 的分数) 。 Alice 的目标是 最大化 分数差,Bob 的目标是 最小化 分数差。
给你一个长度为 n 的整数数组 stones ,其中 stones[i] 是 从左边起 第 i 个石子的价值。请你返回在双方都采用 最优 策略的情况下,Alice 和 Bob 的 分数之差 。
示例 1:
输入:stones = [-1,2,-3,4,-5]
输出:5
解释:

  • Alice 移除最左边的 4 个石子,得分增加 (-1) + 2 + (-3) + 4 = 2 ,并且将一个价值为 2 的石子放在最左边。stones = [2,-5] 。
  • Bob 移除最左边的 2 个石子,得分增加 2 + (-5) = -3 ,并且将一个价值为 -3 的石子放在最左边。stones = [-3] 。
    两者分数之差为 2 - (-3) = 5 。
    示例 2:
    输入:stones = [7,-6,5,10,5,-2,-6]
    输出:13
    解释:
  • Alice 移除所有石子,得分增加 7 + (-6) + 5 + 10 + 5 + (-2) + (-6) = 13 ,并且将一个价值为 13 的石子放在最左边。stones = [13] 。
    两者分数之差为 13 - 0 = 13 。
    示例 3:
    输入:stones = [-10,-12]
    输出:-22
    解释:
  • Alice 只有一种操作,就是移除所有石子。得分增加 (-10) + (-12) = -22 ,并且将一个价值为 -22 的石子放在最左边。stones = [-22] 。
    两者分数之差为 (-22) - 0 = -22 。
    参数范围:
    n == stones.length
    2 <= n <= 105
    -104 <= stones[i] <= 104

分析

思路

dp[i]表示剩余i个石头时,(先手方分数-后手方分数)的最大值。计算dp[i]时,假定移除石头后,还剩j个,也就是总共(包括之前移除)移除m_c-j个。至少移除一个旧石头,故j的取值范围[0,i) 。cur = stones[0,m_c-j)个石头的价值和 - dp[j]。dp[i]等于cur的最大值。

x > 1

初始状态下,只能移除2个,不能移除1个。
非初始状态下,由于必定会移除新石头,所以移除一个旧石头就可以了。
也就是dp[m_c]时m_c-j不能等于1,也就是j不能m_c-1。j无此限制的取值范围是[0,m_c),加上此限制后就变成[0,m_c-1),即i < m_c-1

注意

题意:包括新石头,只剩一个石头的时候结束。我的理解:不包括新石头,没石头的时候结束。初始状态外,一定有新石头,所以两种等价。初始状态,且石头大于1时,两者等价,都是未结束。一个石头,两者不等价。但本题石头数>=2。所以在本题范围内等价。

怀疑

这个题目可能出错了,可能是不放新石头。这样需要技巧合并i。

代码

错误代码

错误原因:忽略了x>1。
class Solution {
public:
int stoneGameVIII(vector& stones) {
m_c = stones.size();
vector dp(m_c + 1);//dp[i]表示剩余i个石头时,(先手方分数-后手方分数)的最大值
//计算dp[i]时,假定移除石头后,还剩j个,也就是移除m_c-j个
// cur = stones[0,m_c-j)个石头的价值和 - dp[j]
vector vSum = { 0 };
for (const auto& n : stones)
{
vSum.emplace_back(n + vSum.back());
}
int iMax = vSum[m_c - 0]-dp[0];
for (int i = 1; i <= m_c; i++)
{
dp[i] = iMax;
iMax = max(iMax, vSum[m_c - i] - dp[i]);
}
return dp.back();
}
int m_c;
};

修正缺陷后

解决方法

class Solution {
public:
	int stoneGameVIII(vector<int>& stones) {
		m_c = stones.size();
		//dp[i]表示剩余i个石头时,(先手方分数-后手方分数)的最大值
		m_dp.resize(m_c + 1);
		//计算dp[i]时,假定移除石头后,还剩j个,也就是移除m_c-j个
		// j的取值范围[0,i) 且m_c-j>1
		// cur = stones[0,m_c-j)个石头的价值和 - dp[j]
		vector<int> vSum = { 0 };
		for (const auto& n : stones)
		{
			vSum.emplace_back(n + vSum.back());
		}
		int iMax = vSum[m_c - 0]-m_dp[0];
		for (int i = 1; i <= m_c; i++)
		{
			m_dp[i] = iMax;
			if (m_c - i > 1)
			{
				iMax = max(iMax, vSum[m_c - i] - m_dp[i]);
			}
		}
		return m_dp.back();
	}
	int m_c;
	vector<int> m_dp;//dp[i]表示剩余i个石头时,(先手方分数-后手方分数)的最大值
};

测试用例

template
void Assert(const vector& v1, const vector& v2)
{
if (v1.size() != v2.size())
{
assert(false);
return;
}
for (int i = 0; i < v1.size(); i++)
{
assert(v1[i] == v2[i]);
}
}

template
void Assert(const T& t1, const T& t2)
{
assert(t1 == t2);
}

int main()
{
Solution slu;
vector stones;
int res = 0;
stones = { -1, 2, -3, 4, -5, 6 };
res = slu.stoneGameVIII(stones);
Assert(3, res);
Assert(vector{0, 3, 3, 3, 3, 3, 3}, slu.m_dp);
stones = { -3,-5,3 };
res = slu.stoneGameVIII(stones);
Assert(-3, res);
Assert(vector{0, -5,-3,-3}, slu.m_dp);
stones = { -1, 2, -3, 4, -5 };
res = slu.stoneGameVIII(stones);
Assert(5, res);
Assert(vector{0, -3, 5, 5, 5, 5}, slu.m_dp);
stones = { -10,-12 };
res = slu.stoneGameVIII(stones);
Assert(-22, res);
Assert(vector{0, -22,-22}, slu.m_dp);

//CConsole::Out(res);

}

2023年2月旧代码

class Solution {
public:
int stoneGameVIII(vector& stones) {
m_c = stones.size();
vector preSum;
int iSum = 0;
for (const auto& s : stones)
{
iSum += s;
preSum.push_back(iSum);
}
vector dp(m_c);
dp.back() = preSum.back();
for (int i = m_c - 2; i >= 1; i–)
{
dp[i] = max(dp[i + 1], preSum[i] - dp[i + 1]);
}
return dp[1];
}
int m_c;
};

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771

如何你想快速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

相关下载

想高屋建瓴的学习算法,请下载《闻缺陷则喜算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653

| 鄙人想对大家说的话
|
|-|
|闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。|
| 墨家名称的来源:有所得以墨记之。 |
|如果程序是一条龙,那算法就是他的是睛|

测试环境

操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 C++17

C++前缀和算法的应用:石头游戏 VIII 原理源码测试用例,# 算法题,数据结构与算法,1024程序员节,c++,算法,前缀和,石头游戏,测试用例,leetcode文章来源地址https://www.toymoban.com/news/detail-714562.html

到了这里,关于C++前缀和算法的应用:石头游戏 VIII 原理源码测试用例的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • C++前缀和算法的应用:统计上升四元组

    C++前缀和算法的应用:统计上升四元组 C++算法:前缀和、前缀乘积、前缀异或的原理、源码及测试用例 包括课程视频 给你一个长度为 n 下标从 0 开始的整数数组 nums ,它包含 1 到 n 的所有数字,请你返回上升四元组的数目。 如果一个四元组 (i, j, k, l) 满足以下条件,我们称

    2024年02月06日
    浏览(45)
  • C++前缀和算法:构造乘积矩阵

    C++算法:前缀和基础 给你一个下标从 0 开始、大小为 n * m 的二维整数矩阵 grid ,定义一个下标从 0 开始、大小为 n * m 的的二维矩阵 p。如果满足以下条件,则称 p 为 grid 的 乘积矩阵 : 对于每个元素 p[i][j] ,它的值等于除了 grid[i][j] 外所有元素的乘积。乘积对 12345 取余数。

    2024年02月08日
    浏览(49)
  • C++基础算法前缀和和差分篇

    📟作者主页:慢热的陕西人 🌴专栏链接:C++算法 📣欢迎各位大佬👍点赞🔥关注🚓收藏,🍉留言 主要讲解了前缀和和差分算法 Ⅵ .Ⅰ前缀和 ① 一维前缀和 : ​ 就是构建一个新的数组 s ,用来存储另一个数组的和前 i 个数组元素的和。用公式表达就是: S [ i ] = a [ 0 ]

    2024年02月16日
    浏览(52)
  • 【0基础学算法】前缀和 (超详细讲解+私人笔记+源码)

    目录 什么是前缀和 我们为什么要去学习前缀和 前缀和实现 如何求s[i]  S[i]的作用  模板 一维前缀和 二维前缀和 题目 第一题 第二题 哈喽大家好,很长时间忘记更新咱们的学算法系列文章了,今天我们终于迎来了我们零基础学算法的第四期内容,那就是前缀和,前缀和其实

    2024年02月05日
    浏览(66)
  • 一文带你深入了解算法笔记中的前缀与差分(附源码)

    📖作者介绍:22级树莓人(计算机专业),热爱编程<目前在c++阶段, 因为最近参加新星计划算法赛道(白佬),所以加快了脚步,果然急迫感会增加动力 ——目标Windows,MySQL,Qt,数据结构与算法,Linux,多线程,会持续分享学习成果和小项目的 📖作者主页:热爱编程的

    2023年04月12日
    浏览(75)
  • 石头剪刀步微信小程序游戏

    之前接了学弟的一个课程作业,但是因为某些原因,最终换成了一个新的爬虫项目。 这个作业就是一个石头剪刀步的微信小游戏。就是与系统随机的 单机 PK,暂时没有去做联接的,没有那个云服务器,但是可以用轮讯的方式去多人联机玩儿。内容虽小但是传统的基本框架都

    2024年02月07日
    浏览(38)
  • Python语法小游戏——石头、剪刀、布

    一个有趣的小游戏——石头、剪刀、布 要求如下: 1、键盘输入 (1 石头 2 剪刀 3 布) 2、电脑随机产生(1 石头 2 剪刀 3 布) 3、输出 划拳的结果。 可以锻炼简单的思维逻辑能力,首先,我们需要知道要用到什么。 结尾附完整代码 第一步,我们需要获取用户输入信息,再随

    2024年02月04日
    浏览(33)
  • python代码练习:石头剪刀布猜拳游戏

    使用Python实现人机石头剪刀布猜拳小游戏,并且最后能够统计分数和局数

    2024年02月12日
    浏览(28)
  • 【算法思考记录】【前缀和,C++】力扣1277. 统计全为 1 的正方形子矩阵

    原题链接 题目要求我们统计在一个由0和1构成的矩阵中,所有完全由1组成的正方形子矩阵的数量。这是一道中等难度的算法题目,其关键在于高效地计算出不同大小的正方形子矩阵是否完全由1组成。 解决此问题的一个有效方法是使用 前缀和 算法。前缀和是一种预处理技术

    2024年02月03日
    浏览(39)
  • 华为OD机试 - 石头剪刀布游戏(Java & JS & Python & C)

    题目描述 石头剪刀布游戏有 3 种出拳形状:石头、剪刀、布。分别用字母A、B、C表示。 游戏规则: 出拳形状之间的胜负规则如下: A B; B C; C A; \\\"\\\" 左边一个字母,表示相对优势形状。右边一个字母,表示相对劣势形状。   当本场次中有且仅有一种出拳形状优于其他出拳

    2024年02月04日
    浏览(52)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包