leetcode biweekly contest 101
第四题比较有意思的题目,希望多多来点有意思的题目,模板题目太多了。
6354. K 件物品的最大和
袋子中装有一些物品,每个物品上都标记着数字 1
、0
或 -1
。
给你四个非负整数 numOnes
、numZeros
、numNegOnes
和 k
。
袋子最初包含:
numOnes
件标记为1
的物品。numZeroes
件标记为0
的物品。numNegOnes
件标记为-1
的物品。
现计划从这些物品中恰好选出 k
件物品。返回所有可行方案中,物品上所标记数字之和的最大值。
示例 1:
输入:numOnes = 3, numZeros = 2, numNegOnes = 0, k = 2 |
示例 2:
输入:numOnes = 3, numZeros = 2, numNegOnes = 0, k = 4 |
提示:
0 <= numOnes, numZeros, numNegOnes <= 50
0 <= k <= numOnes + numZeros + numNegOnes
地址
https://leetcode.cn/contest/weekly-contest-338/problems/k-items-with-the-maximum-sum/
题意
直接模拟
思路
- 直接模拟即可,我们知道最小的数字要么为一位,要么为两位;
- 如果两个数组含有相同的数字,则直接返回相同数字中的最小一位即可;
- 如果两个数组中不含有相同的数字,则此时我们需要从数组 $1$ 中选择最小一位,同时从数组 $2$ 中选择最小的一位,然后二者构成最小的数字即可;
- 复杂度分析:
- 时间复杂度:$O(n^2)$。实际可以优化到 $n\log n$。
- 空间复杂度:$O(n)$。
代码
class Solution { |
2606. 找到最大开销的子字符串
给你一个字符串 s
,一个字符 互不相同 的字符串 chars
和一个长度与 chars
相同的整数数组 vals
。
子字符串的开销 是一个子字符串中所有字符对应价值之和。空字符串的开销是 0
。
字符的价值 定义如下:
- 如果字符不在字符串
chars
中,那么它的价值是它在字母表中的位置(下标从1 开始)。- 比方说,
'a'
的价值为1
,'b'
的价值为2
,以此类推,'z'
的价值为26
。
- 比方说,
- 否则,如果这个字符在
chars
中的位置为i
,那么它的价值就是vals[i]
。
请你返回字符串 s
的所有子字符串中的最大开销。
示例 1:
输入:s = "adaa", chars = "d", vals = [-1000] |
示例 2:
输入:s = "abc", chars = "abc", vals = [-1,-1,-1] |
提示:
1 <= s.length <= 105
s
只包含小写英文字母。1 <= chars.length <= 26
chars
只包含小写英文字母,且 互不相同 。vals.length == chars.length
-1000 <= vals[i] <= 1000
地址
https://leetcode.cn/contest/biweekly-contest-101/problems/find-the-substring-with-maximum-cost/
题意
前缀和
思路
- 题目为简单的贪心法则,由于只能计算连续的字符串的开销,因此我们计算当前从 $0$ 开始到 $i$ 的开销,即 $s[0\cdots i]$ 的开销,每次找到 $s[0\cdots j],j < k$ 的最小值,用当前值减去最小值即可找到以 $i$ 为结尾的连续子字符串的最大开销;
- 复杂度分析:
- 时间复杂度:$O(n \log n)$,其中 $n$ 为字符串的长度。
- 空间复杂度:$O(n)$,其中 $n$ 为字符串的长度。
代码
class Solution { |
2607. 使子数组元素和相等
给你一个下标从 0 开始的整数数组 arr
和一个整数 k
。数组 arr
是一个循环数组。换句话说,数组中的最后一个元素的下一个元素是数组中的第一个元素,数组中第一个元素的前一个元素是数组中的最后一个元素。
你可以执行下述运算任意次:
- 选中
arr
中任意一个元素,并使其值加上1
或减去1
。
执行运算使每个长度为 k
的 子数组 的元素总和都相等,返回所需要的最少运算次数。
子数组 是数组的一个连续部分。
示例 1:
输入:arr = [1,4,1,3], k = 2 |
示例 2:
输入:arr = [2,5,5,7], k = 3 |
提示:
1 <= k <= arr.length <= 105
1 <= arr[i] <= 109
地址
https://leetcode.cn/contest/biweekly-contest-101/problems/make-k-subarray-sums-equal/
题意
中位数
思路
首先我们需要将数组进行分类筛选,若使得长度为 $k$ 子数组和相等,根据公式可以知道:
$$ \sum_{j=i}^{j+k-1}arr[j] = \sum_{i=i+1}^{j+k}arr[j]$$
由上述公式可以推出 $arr[i] = arr[(i + k) \mod n]$;
我们只需要满足所有的 $arr[i] = arr[(i + k) \mod n]$,则一定可以使得所有长度为 $k$ 的连续子数组相等,因此题目变成了使得$arr[i] = arr[(i + k) \mod n]$ 相等的最少运算次数,此时题目就变得很简单了。
首先我们按照上述规则对需要相等的元素进行分类,此时我们可以模仿欧拉素数筛选法,很快即可对数组中的元素进行分类;再次思考一个经典问题,如何使得数组中元素相等的最小操作,实际是将数组中的元素全部变为当前数组的中位数时所需的操作数最小,此时我们直接求出所有元素与中位数的差的绝对值之和即可。
复杂度分析:
- 时间复杂度:$O(n \log n)$,其中 $n$ 为数组的长度。
- 空间复杂度:$O(n)$,其中 $n$ 为数组的长度。
代码
class Solution { |
2608. 图中的最短环
现有一个含 n
个顶点的 双向 图,每个顶点按从 0
到 n - 1
标记。图中的边由二维整数数组 edges
表示,其中 edges[i] = [ui, vi]
表示顶点 ui
和 vi
之间存在一条边。每对顶点最多通过一条边连接,并且不存在与自身相连的顶点。
返回图中 最短 环的长度。如果不存在环,则返回 -1
。
环 是指以同一节点开始和结束,并且路径中的每条边仅使用一次。
示例 1:
输入:n = 7, edges = [[0,1],[1,2],[2,0],[3,4],[4,5],[5,6],[6,3]] |
示例 2:
输入:n = 4, edges = [[0,1],[0,2]] |
提示:
2 <= n <= 1000
1 <= edges.length <= 1000
edges[i].length == 2
0 <= ui, vi < n
ui != vi
- 不存在重复的边
地址
https://leetcode.cn/contest/biweekly-contest-101/problems/shortest-cycle-in-a-graph/
题意
广度优先搜索
思路
- 题目比较有意思,一开始我还想着是不是需要弄
tarjan
找连通单元,后来发现不需要,直接枚举每点为起点,找到该点为起点的最小环即可。 - 找到环需要一点技巧,以 $i$ 为起点找到所有点到达 $i$ 的最小距离,如果发现存在环则记录环的大小即可,即对于同一个点,存在两条不同的路径到达该点时,即存在环,我们使用
bfs
即可,此时可以模仿dfs
记录父亲节点,放置出现重复。 - 看到有人还有另外一种做法,即删除边的做法,每次删除边 $(i,j)$ 后,如果任然存在 $i$ 到 $j$ 的路径,则表示 $i$ 到 $j$ 之间一定存在环,这个解法非常容易理解,也比较好写,推荐该解法。
- 复杂度分析:
- 时间复杂度:$O(n^2)$,其中$ n$ 表示节点的数目。
- 空间复杂度:$O(n+m)$,其中$ n$ 表示节点的数目,$m$ 表示边数。
代码
- 删除点
class Solution { |
- 删除边
class Solution { |
欢迎关注和打赏,感谢支持!
- 关注我的博客: http://whistle-wind.com/
- 关注我的知乎:https://www.zhihu.com/people/da-hua-niu
- 关注我的微信公众号: 公务程序猿
扫描二维码,分享此文章