1561. 你可以获得的最大硬币数目 #贪心

题目链接

题意

有 3n 堆数目不一的硬币,你和你的朋友们打算按以下方式分硬币:

  • 每一轮中,你将会选出 任意 3 堆硬币(不一定连续)。
  • Alice 将会取走硬币数量最多的那一堆。
  • 你将会取走硬币数量第二多的那一堆。
  • Bob 将会取走最后一堆。

重复这个过程,直到没有更多硬币。

给你一个整数数组pilespiles[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 题解的更多相关文章

  1. Leetcode 周赛#202 题解

    本周的周赛题目质量不是很高,因此只给出最后两题题解(懒). 1552 两球之间的磁力 #二分答案 题目链接 题意 有n个空篮子,第i个篮子位置为position[i],现希望将m个球放到这些空篮子,使 ...

  2. LeetCode周赛#204 题解

    1566. 重复至少 K 次且长度为 M 的模式 #模拟 题目链接 题意 给定正整数数组 arr,请你找出一个长度为 m 且在数组中至少重复 k 次的模式. 模式 是由一个或多个值组成的子数组(连续的 ...

  3. Leetcode 周赛#201 题解

    1545 找出第N个二进制字符串的第K位 #分治 题目链接 题意 给定正整数\(n(\leq 20)\)与\(k\),二进制串\(S_n\)形成规则有: \(S_1 = "0"\) ...

  4. Leetcode 周赛#200 题解

    1535 找出数组游戏的赢家 #模拟+优化 题目链接 题意 给你一个由 不同 整数组成的整数数组 arr 和一个整数 k(\(1\leq k\leq1e9\)) .每回合游戏都在数组的arr[0] 和 ...

  5. Leetcode】周赛203 查找大小为M的最新分组

    题意: 给你一个数组 arr ,该数组表示一个从 1 到 n 的数字排列.有一个长度为 n 的二进制字符串,该字符串上的所有位最初都设置为 0 . 在从 1 到 n 的每个步骤 i 中(假设二进制字符 ...

  6. LeetCode双周赛#33 题解

    5480. 可以到达所有点的最少点数目 #贪心 题目链接 题意 给定有向无环图,编号从0到n-1,一个边集数组edges(表示从某个顶点到另一顶点的有向边),现要找到最小的顶点集合,使得从这些点出发, ...

  7. Leetcode 双周赛#32 题解

    1540 K次操作转变字符串 #计数 题目链接 题意 给定两字符串\(s\)和\(t\),要求你在\(k\)次操作以内将字符串\(s\)转变为\(t\),其中第\(i\)次操作时,可选择如下操作: 选 ...

  8. 【Leetcode周赛】从contest-111开始。(一般是10个contest写一篇文章)

    Contest 111 (题号941-944)(2019年1月19日,补充题解,主要是943题) 链接:https://leetcode.com/contest/weekly-contest-111 ...

  9. 【Leetcode周赛】从contest-41开始。(一般是10个contest写一篇文章)

    Contest 41 ()(题号) Contest 42 ()(题号) Contest 43 ()(题号) Contest 44 (2018年12月6日,周四上午)(题号653—656) 链接:htt ...

随机推荐

  1. Mybatis---06Mybatis配置文件浅析(四)

    参考链接:深入理解Mybatis插件开发 1.plugins:与其称为Mybatis插件,不如叫Mybatis拦截器,更加符合其功能定位,实际上它就是一个拦截器,应用代理模式,在方法级别上进行拦截. ...

  2. 解决Linux-Centos7启动Mysql服务失败丢失mysql.sock问题

    在新安装mysql后进行启动发现报错 mysql启动服务命令 systemctl start mysqld@3306 Starting mysqld (via systemctl):  Job for ...

  3. 如何给office2019安装visio2019

    1.关于版本,如果你的office2019推荐安装2019visio,2019office只有安装2019visio才不会互斥 下载地址是百度网盘传送,也要安装教程,可是里面的激活器用不了 https ...

  4. 深度对比Apache CarbonData、Hudi和Open Delta三大开源数据湖方案

    摘要:今天我们就来解构数据湖的核心需求,同时深度对比Apache CarbonData.Hudi和Open Delta三大解决方案,帮助用户更好地针对自身场景来做数据湖方案选型. 背景 我们已经看到, ...

  5. 4G DTU的应用场景介绍

    4G DTU因为信号要比传统的gprs网络要好,目前已经被广泛应用于物联网产业链中的M2M行业,以远向4G DTU LTE-520为例,它的应用场景如智能电网.智能交通.智能家居.金融.移动 POS ...

  6. pause容器作用

    1.概念 Pause容器 全称infrastucture container(又叫infra)基础容器. 作为init pod存在,其他pod都会从pause 容器中fork出来. 每个Pod里运行着 ...

  7. vim实现CTRL+S为保存快捷键

    用vim正撸代码撸的飞起,突然Xshell就掉线了,真是太蛋疼了. 于是开始怀念起vs下撸代码时随时随地ctrl+s保存的快捷键,百度了一下,网上的vim实现ctrl+s保存的快捷键设置都有问题,自己 ...

  8. How to Convert and Import VHD to VMDK (VMWare)

    VHD or Virtual Hard Disk is the disk image format used by Microsoft virtualization software such as ...

  9. 常用的Linux命令,日常收集记录

    1.# yum install -y xxxx 解释:install代表往系统中安装一个或者多个软件包:-y 代表回答全部问题为是 2.# ps -ef | grep yum   (根据进程名来查看进 ...

  10. Thinkphp3.2 cms之权限管理

    五.权限管理 <?php namespace Admin\Controller; use Think\Controller; class CommonController extends Con ...