# Leetcode 第278场周赛题解

# Problem A - 将找到的值乘以 2 (opens new window)

# 方法一:模拟

按要求模拟即可。

  • 时间复杂度 O(N+logM)\mathcal{O}(N+\log M)
  • 空间复杂度 O(N)\mathcal{O}(N)
参考代码(Python 3)
class Solution:
    def findFinalValue(self, nums: List[int], original: int) -> int:
        s = set(nums)
        while original in s:
            original *= 2
        return original
1
2
3
4
5
6

# Problem B - 分组得分最高的所有下标 (opens new window)

# 方法一:前缀和

遍历过程中,维护前缀中 0 的个数和后缀中 1 的个数。

  • 时间复杂度 O(N)\mathcal{O}(N)
  • 空间复杂度 O(N)\mathcal{O}(N)
参考代码(Python 3)
class Solution:
    def maxScoreIndices(self, nums: List[int]) -> List[int]:
        n = len(nums)
        r = nums.count(1)
        l = 0
        hi = r
        ans = [0]
        for i in range(n):
            if nums[i] == 1:
                r -= 1
            if nums[i] == 0:
                l += 1
            now = l + r
            if now > hi:
                hi = now
                ans = []
            if now == hi:
                ans.append(i + 1)
        return ans
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

# Problem C - 查找给定哈希值的子串 (opens new window)

# 方法一:倒序滑动窗口

长度 KK 是固定的,这提示我们使用滑动窗口来计算子串的哈希值。这里我们有两种选择:

  • 从左向右滑动
  • 从右向左滑动

但我们发现,如果从左向右滑动,在更新哈希值时,我们需要进行同余除法。但本题中的模数MM 不一定是质数,逆元未必存在。

因此我们只能从右向左滑动。

一个小细节:记录中间结果的时候并不需要记录下完整的子串,只需要记录开始(或结束)位置即可。否则,如果有大量相同子串且都为答案,而使用的语言底层对字符串切片没有相应优化的话,就会导致超时。

  • 时间复杂度 O(N)\mathcal{O}(N)
  • 空间复杂度 O(N)\mathcal{O}(N)
参考代码(Python 3)
class Solution:
    def subStrHash(self, s: str, power: int, modulo: int, k: int, hashValue: int) -> str:
        n = len(s)
        pk = 1
        
        now = 0
        for i in range(n - k, n):
            c = ord(s[i]) - ord('a') + 1
            now = (now + c * pk) % modulo
            if i != n - 1:
                pk = pk * power % modulo
            
        ans = -1 if now != hashValue else n - k
        
        for i in range(n - k - 1, -1, -1):
            c = ord(s[i]) - ord('a') + 1
            d = ord(s[i + k]) - ord('a') + 1
            now += modulo - d * pk % modulo
            now = (now * power + c) % modulo
            if now == hashValue:
                ans = i
        
        return s[ans:ans + k]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23

# Problem D - 字符串分组 (opens new window)

# 方法一:并查集+枚举

本题题面存在一定问题,结果并不唯一。需要加上限制保证每个分组内的连通性,才能使得答案唯一。下面的分析假设已经有这一限制。

本题中我们需要关注的是字符集合,且题目说了每个字符串中都不包含相同的字母。因此我们可以将每个字符串转为对应的一个二进制数。

下面考虑三种操作:

  • 删除:相当于把某一位从 1 变为 0
  • 添加:相当于把某一位从 0 变为 1
  • 修改:相当于把一个 0 变为 1, 同时把一个 1 变为 0(不考虑保持不变的情况)

从而删除和添加可以合并为一种操作,也即修改一位的操作;而修改则对应于修改两位的操作。当然,还可以保持不变。

题目要求的分组关系实际上就是并查集。我们遍历所有字符串(对应的二进制数),然后枚举所有可能的操作,判断操作后的结果是否已经在之前的遍历中出现过。如果已经出现过,则在并查集中连边。

最后统计并查集中每个连通分量的信息即可得到答案。

  • 时间复杂度O(NΣ2α(N))\mathcal{O}(N|\Sigma|^2\alpha(N))
  • 空间复杂度O(N)\mathcal{O}(N)
参考代码(C++)
struct UnionFind {
  int n;
  vector<int> parent, size;

  UnionFind(int n) {
    this->n = n;
    parent = vector<int>(n);
    size = vector<int>(n, 1);
    for (int i = 0; i < n; ++i)
      parent[i] = i;
  }

  int find(int idx) {
    if (parent[idx] == idx)
      return idx;
    return parent[idx] = find(parent[idx]);
  }

  void connect(int a, int b) {
    int fa = find(a), fb = find(b);
    if (fa != fb) {
      if (size[fa] > size[fb]) {
        parent[fb] = fa;
        size[fa] += size[fb];
      } else {
        parent[fa] = fb;
        size[fb] += size[fa];
      }
    }
  }
};

class Solution {
public:
    vector<int> groupStrings(vector<string>& words) {
        int n = words.size();
        vector<int> bits(n);
        for (int i = 0; i < n; ++i)
            for (char c : words[i])
                bits[i] ^= 1 << (c - 'a');
        
        UnionFind uf(n);
        unordered_map<int, int> mp;
        for (int i = 0; i < n; ++i) {
            if (mp.count(bits[i]))
                uf.connect(mp[bits[i]], i);
            else {
                for (int j = 0; j < 26; ++j) {
                    int nb = bits[i] ^ (1 << j);
                    if (mp.count(nb))
                        uf.connect(mp[nb], i);
                }
                for (int j = 0; j < 26; ++j) {
                    int mskj = 1 << j;
                    if ((bits[i] & mskj) == 0)
                        continue;
                    for (int k = 0; k < 26; ++k) {
                        int mskk = 1 << k;
                        if (bits[i] & mskk)
                            continue;
                        int nb = bits[i] ^ mskj ^ mskk;
                        if (mp.count(nb))
                            uf.connect(mp[nb], i);
                    }
                }
            }
            
            if (!mp.count(bits[i]))
                mp[bits[i]] = i;
        }
        
        int groups = 0, largest = 0;
        for (int i = 0; i < n; ++i) {
            if (uf.find(i) == i) {
                groups++;
                largest = max(largest, uf.size[i]);
            }
        }
        
        return {groups, largest};
    }
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82