LeetCode周赛#203 题解
1561. 你可以获得的最大硬币数目 #贪心
题目链接
题意
有 3n 堆数目不一的硬币,你和你的朋友们打算按以下方式分硬币:
- 每一轮中,你将会选出 任意 3 堆硬币(不一定连续)。
- Alice 将会取走硬币数量最多的那一堆。
- 你将会取走硬币数量第二多的那一堆。
- Bob 将会取走最后一堆。
重复这个过程,直到没有更多硬币。
给你一个整数数组piles
,piles[i]
是第 i
堆中硬币的数目。现要你求出可获得的最大硬币数目。
分析
毫无疑问,需要先将硬币堆降序排序。既然每一轮我只能拿当前三堆中的第二大,为了让我的硬币数更多,应先让Bob从后面堆中取极小硬币堆,我与Alice在前面取极大硬币堆。
class Solution {
public:
int maxCoins(vector<int>& piles) {
sort(piles.begin(), piles.end());
reverse(piles.begin(), piles.end());
int cnt = 0, sum = 0;
int L = 0, R = piles.size() - 1;
for (int L = 1; L < piles.size() && L < R; L += 2, R--){
sum += piles[L];
}
return sum;
}
};
1562. 查找大小为M的最新分组 #二分思想 #STL #逆向思维
题目链接
题意
给定一数组 arr
,该数组表示一个从 1 到 n 的数字排列。有一个长度为 n 的二进制字符串,该字符串上的所有位最初都设置为 0 。在从 1 到 n 的每个步骤 i
中(从 1 开始索引),二进制字符串上位于位置 arr[i]
的位将会设为 1 。
给定整数 m ,现要找出二进制字符串上存在长度为 m的一组 "1"
的最后步骤。一组 "1"
是一个连续的、由 1 组成的子串,且左右两边不再有可以延伸的 1 。如果不存在这样的步骤,请返回 -1 。
样例
分析
\(O(n)\)的解法可戳此,此处提供易写的版本,同时学习set
的 用法,不过复杂度约为\(O(nlogn)\)。
既然它要我们找出“最后”的步骤,可以尝试逆向思考,即从全为"111…11"
串转化为"110010..."
串,一旦串中出现长度恰为m的"1"子串即可返回。
如何在每一次操作中计算每个"1"串的长度?我们可以利用set
的自动排序以及它自带的二分函数,将每一次置0的位置插入到set
中,插入时需要通过二分(尽量使用set
自带的二分函数,因为通用的二分函数不支持随机访问的容器,使得复杂度达到\(O(n)\)),主要是为了找到它的新插入位置附近的迭代器,通过对迭代器代表的位置做差即可得到特定"1"串的长度。
class Solution {
private:
int sum[505] = {0};
set<int> vis;
public:
int findLatestStep(vector<int>& arr, int m) {
int len = arr.size();
if(len == m) return m;
vis.emplace(0); //相比于insert(),emplace()直接构造对象,效率更高
vis.emplace(len + 1);
for (int i = len - 1; i >= 0; i--){
auto it = vis.upper_bound(arr[i]); //返回第一个大于arr[i]的元素迭代器
int L = *prev(it), R = *it; //prev()获取一个距离指定迭代器 n 个元素的迭代器,n取正数时,向左移动。
if(arr[i] - L - 1 == m || R - arr[i] - 1 == m) //注意,要-1,因为是两个0位之差
return i;
vis.emplace(arr[i]); //分裂成两堆
}
return -1;
}
};
1563. 石子游戏 #记忆化搜索
题目链接
题意
几块石子排成一行 ,每块石子都有一个关联值,关联值为整数,由数组stoneValue
给出。
游戏中的每一轮:Alice 会将这行石子分成两个 非空行(长度不一定相等);Bob 负责计算每一行的值,即此行中所有石子的值的总和。Bob 会丢弃值最大的行,Alice 的得分为剩下那行的值(每轮累加)。如果两行的值相等,Bob 让 Alice 决定丢弃哪一行。下一轮从剩下的那一行开始。当只剩下一块石子时,游戏结束。Alice 的分数最初为 0 。
现要求 Alice 能够获得的最大分数 。
样例
分析
通过记忆化搜索来模拟即可,转移方程分三种情况,见下方代码。
class Solution {
private:
int sum[505] = {0}, dp[505][505] = {0};
int len;
public:
int dfs(int lo, int hi){
int mymax = 0;
if(lo <= 0 || hi > len) return 0;
if(hi - lo <= 0) return 0;
if(dp[lo][hi] > 0) return dp[lo][hi];
for (int i = lo; i <= hi; i++){
int presum = sum[i] - sum[lo - 1], latsum = sum[hi] - sum[i];
if(presum == latsum)
mymax = max(mymax, max(dfs(lo, i), dfs(i + 1, hi)) + presum);
else if(presum > latsum)
mymax = max(mymax, dfs(i + 1, hi) + latsum);
else
mymax = max(mymax, dfs(lo, i) + presum);
}
return dp[lo][hi] = mymax;
}
int stoneGameV(vector<int>& stoneValue) {
len = stoneValue.size();
if(len == 1) return 0;
if(len == 2) return min(stoneValue[0], stoneValue[1]);
memset(dp, -1, sizeof(dp));
for (int i = 0; i < len; i++) sum[i + 1] = sum[i - 1 + 1] + stoneValue[i];
return dfs(1, len);
}
};
LeetCode周赛#203 题解的更多相关文章
- Leetcode 周赛#202 题解
本周的周赛题目质量不是很高,因此只给出最后两题题解(懒). 1552 两球之间的磁力 #二分答案 题目链接 题意 有n个空篮子,第i个篮子位置为position[i],现希望将m个球放到这些空篮子,使 ...
- LeetCode周赛#204 题解
1566. 重复至少 K 次且长度为 M 的模式 #模拟 题目链接 题意 给定正整数数组 arr,请你找出一个长度为 m 且在数组中至少重复 k 次的模式. 模式 是由一个或多个值组成的子数组(连续的 ...
- Leetcode 周赛#201 题解
1545 找出第N个二进制字符串的第K位 #分治 题目链接 题意 给定正整数\(n(\leq 20)\)与\(k\),二进制串\(S_n\)形成规则有: \(S_1 = "0"\) ...
- Leetcode 周赛#200 题解
1535 找出数组游戏的赢家 #模拟+优化 题目链接 题意 给你一个由 不同 整数组成的整数数组 arr 和一个整数 k(\(1\leq k\leq1e9\)) .每回合游戏都在数组的arr[0] 和 ...
- Leetcode】周赛203 查找大小为M的最新分组
题意: 给你一个数组 arr ,该数组表示一个从 1 到 n 的数字排列.有一个长度为 n 的二进制字符串,该字符串上的所有位最初都设置为 0 . 在从 1 到 n 的每个步骤 i 中(假设二进制字符 ...
- LeetCode双周赛#33 题解
5480. 可以到达所有点的最少点数目 #贪心 题目链接 题意 给定有向无环图,编号从0到n-1,一个边集数组edges(表示从某个顶点到另一顶点的有向边),现要找到最小的顶点集合,使得从这些点出发, ...
- Leetcode 双周赛#32 题解
1540 K次操作转变字符串 #计数 题目链接 题意 给定两字符串\(s\)和\(t\),要求你在\(k\)次操作以内将字符串\(s\)转变为\(t\),其中第\(i\)次操作时,可选择如下操作: 选 ...
- 【Leetcode周赛】从contest-111开始。(一般是10个contest写一篇文章)
Contest 111 (题号941-944)(2019年1月19日,补充题解,主要是943题) 链接:https://leetcode.com/contest/weekly-contest-111 ...
- 【Leetcode周赛】从contest-41开始。(一般是10个contest写一篇文章)
Contest 41 ()(题号) Contest 42 ()(题号) Contest 43 ()(题号) Contest 44 (2018年12月6日,周四上午)(题号653—656) 链接:htt ...
随机推荐
- 【总结】IP
一. IP基本认识 1.IP 在 TCP/IP 参考模型中处于第三层,也就是网络层. 网络层的主要作用是:实现主机与主机之间的通信,也叫点对点(end to end)通信 2.网络层与数据链路层有什么 ...
- Luogu P4957 [COCI2017-2018#6] Alkemija
题意 有 \(n\) 种已知物质,现在手上有 \(m\) 种,每种无限多个.已知 \(k\) 种反应,每种可以将一些反应物变成一些生成物.求经过这些反应过后最多可以有多少种不同的物质. \(\text ...
- P5958 【[POI2017]Sabotaż】
P5958 [[POI2017]Sabotaż] 题意描述 在一棵以1号节点为根节点的树上,有很多纯洁的白点, BUT,突然有一个黑点出现(可能在任意位置) 它要染黑尽可能多的节点,而在一棵子树中, ...
- 测试流程规范--测试准入、准出、停止标准、bug优先级定义
一.背景 最近在推进组内流程规范专项建设,从"研发测试流程"."提测规范"."测试准入标准"."bug优先级标准".&q ...
- JavaMail 发送邮件出现 Connection reset 问题
问题描述 使用 java mail 发送邮件的时候,申请的 163 邮箱作为发件箱,然无论如何配置,均出现 Connection reset,无法正常发送邮件. Exception in thread ...
- php数据映射到echarts中国地图
echarts和php结合 根据php后台数据,映射到地图上所有的省份,地图市下钻的形式,每次下钻到新地图不合并(合并缩放的话会导致下钻地图位置找不到了),添加了自定义工具栏里面的返回到顶级下钻地图 ...
- 16flask错误处理
1,A secret key is required to use CSRF 使用CSRF需要一个密钥,也就是说没有设置秘钥
- JS函数命名规范
语法规范: 任何合法的javascript标识符都可以作为函数的名称. 约定俗成的内容:(非ECMAScript语法,但是为了便于开发者理解和识别,约定的函数命名规范.) 命名方法: 小驼峰式命名法 ...
- 利用Github Action和.Net 5 自动执行米游社原神每日签到福利
GenshinDailyHelper 原神的签到福利是需要单独下载APP进行才可以领取,并且每天需要打卡,虽然奖励并不是很可观,但有一些摩拉,食材和可观的经验书累计起来还是挺有吸引力的.可能本身不怎么 ...
- Socket bind系统调用简要分析
主要查看linux kernel 源码:Socket.c 以及af_inet.c文件 1.1 bind分析 #include <sys/types.h> /* See NOTES */#i ...