LeetCode笔记:Weekly Contest 325

Posted Espresso Macchiato

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了LeetCode笔记:Weekly Contest 325相关的知识,希望对你有一定的参考价值。

1. 题目一

给出题目一的试题链接如下:

1. 解题思路

这一题的思路还是很直接的,就是找一下每一个target的单词与startIndex之间的正反距离然后取最小值即可。

如果target不存在,则返回-1。

2. 代码实现

给出python代码实现如下:

class Solution:
    def closetTarget(self, words: List[str], target: str, startIndex: int) -> int:
        n = len(words)
        res = n+1
        for i, w in enumerate(words):
            if w == target:
                res = min(res, (startIndex - i + n) % n, (i - startIndex + n) % n)
        return res if res != n+1 else -1

提交代码评测得到:耗时65ms,占用内存14MB。

2. 题目二

给出题目二的试题链接如下:

1. 解题思路

这一题思路上就是一个滑动窗口。

我们首先只看左侧,只取左边的元素然后令所有元素个数均不少于k。

然后,我们考虑左侧元素每减少一个,需要从右侧补充多少个元素进行补偿,维护这个滑动窗口直至左侧元素清零。

此时,我们返回所有情况下的最小值即可。

2. 代码实现

给出python代码实现如下:

class Solution:
    def takeCharacters(self, s: str, k: int) -> int:
        cnt = defaultdict(int)
        n = len(s)
        i, j = 0, n-1
        while i < n:
            cnt[s[i]] += 1
            i += 1
            if all(cnt[ch] >= k for ch in "abc"):
                break
        if any(cnt[ch] < k for ch in "abc"):
            return -1
        res = i
        while i > 0:
            i -= 1
            cnt[s[i]] -= 1
            while any(cnt[ch] < k for ch in "abc"):
                cnt[s[j]] += 1
                j -= 1
            res = min(res, i + n-1 - j)
        return res

提交代码评测得到:耗时939ms,占用内存14.9MB。

3. 题目三

给出题目三的试题链接如下:

1. 解题思路

这一题思路上就是一个二分法。

我们假设candy之间的价格差距不少于d,然后通过二分法看一下在这种情况下能否构造一个bucket使得其中的candy总数不少于k。然后我们通过二分法找出d可取的最大值即可。

而关于在价格差距为d的情况下能否构造的问题,我们可以先将price进行排序,然后用二分查找的方式依次找到在间隔不少于d的情况下的下一种糖果的价格,然后看看能否取到k种糖果即可。

2. 代码实现

给出python代码实现如下:

class Solution:
    def maximumTastiness(self, price: List[int], k: int) -> int:
        price = sorted(price)
        n = len(price)
        
        def is_possible(d):
            idx = 0
            for _ in range(k):
                if idx >= n:
                    return False
                idx = bisect.bisect_left(price, price[idx] + d, lo=idx)
            return True
        
        i, j = 0, (price[-1] - price[0]) // (k-1) + 1
        while j-i > 1:
            d = (j+i) // 2
            if is_possible(d):
                i = d
            else:
                j = d
        return i

提交代码评测得到:耗时2010ms,占用内存26MB。

4. 题目四

给出题目四的试题链接如下:

1. 解题思路

这一题思路上我其实还是非常暴力的就用了一个动态规划 + 剪枝的方法,也算是面对超时编程了。

动态规划的方法其实很简单了,就不多做展开了,这里就具体讲讲要如何剪枝:

  1. 我们记录下两个组合中的已有元素之和,如果两边的和都已经大于目标值k,那么剩下的元素(不妨设为m个)就可以直接随机分到两个组合当中了,即 2 m 2^m 2m种组合;
  2. 我们对原数组从大到小排个序,优先对大的元素进行分组,这样可以更快速地达到1当中的条件,进而更进一步地实现剪枝。

通过上述两个操作,虽然方法依然不怎么优雅,不过还是可以通过全部的测试样例,甚至耗时还算过得去……

2. 代码实现

给出python代码实现如下:

class Solution:
    def countPartitions(self, nums: List[int], k: int) -> int:
        nums = sorted(nums, reverse=True)
        
        MOD = 10**9 + 7
        s = sum(nums)
        n = len(nums)
        
        @lru_cache(None)
        def dp(idx, s1, s2):
            if s1 >= k and s2 >= k:
                return pow(2, n-idx, MOD)
            elif s1 > s-k or s2 > s-k or idx >= n:
                return 0
            return (dp(idx+1, s1 + nums[idx], s2) + dp(idx+1, s1, s2 + nums[idx])) % MOD
        
        return dp(0, 0, 0)

提交代码评测得到:耗时2569ms,占用内存522.8MB。

以上是关于LeetCode笔记:Weekly Contest 325的主要内容,如果未能解决你的问题,请参考以下文章

LeetCode笔记:Weekly Contest 317

LeetCode笔记:Weekly Contest 288

LeetCode笔记:Weekly Contest 299

LeetCode笔记:Weekly Contest 307

LeetCode笔记:Weekly Contest 325

LeetCode笔记:Weekly Contest 314