LeetCode 90 双周赛

24 阅读4分钟

2451. 差值数组不同的字符串

给你一个字符串数组 words ,每一个字符串长度都相同,令所有字符串的长度都为 n

每个字符串 words[i] 可以被转化为一个长度为 n - 1差值整数数组 difference[i] ,其中对于 0 <= j <= n - 2difference[i][j] = words[i][j+1] - words[i][j] 。注意两个字母的差值定义为它们在字母表中 位置 之差,也就是说 'a' 的位置是 0'b' 的位置是 1'z' 的位置是 25

  • 比方说,字符串 "acb" 的差值整数数组是 [2 - 0, 1 - 2] = [2, -1]

words 中所有字符串 除了一个字符串以外 ,其他字符串的差值整数数组都相同。你需要找到那个不同的字符串。

请你返回 words差值整数数组 不同的字符串。

提示:

  • 3 <= words.length <= 100
  • n == words[i].length
  • 2 <= n <= 20
  • words[i] 只含有小写英文字母。

示例

输入:words = ["adc","wzy","abc"]
输出:"abc"
解释:
- "adc" 的差值整数数组是 [3 - 0, 2 - 3] = [3, -1] 。
- "wzy" 的差值整数数组是 [25 - 22, 24 - 25]= [3, -1] 。
- "abc" 的差值整数数组是 [1 - 0, 2 - 1] = [1, 1] 。
不同的数组是 [1, 1],所以返回对应的字符串,"abc"。

思路

模拟,计算出所有字符串的差值数组,然后找出其中不同的一个即可。

// C++
class Solution {
public:
    string oddString(vector<string>& words) {
        // 计算所有字符串的差值数组
        vector<vector<int>> d;
        for (auto& s : words) {
            vector<int> v;
            for (int i = 1; i < s.size(); i++) v.push_back(s[i] - s[i - 1]);
            d.push_back(v);
        }
        
        // 比较每个差值数组的每个位置, 找到某一个不同的位置
        for (int i = 1; i < d.size(); i++) {
            for (int j = 0; j < d[i].size(); j++) {
                if (d[i][j] != d[i - 1][j]) {
                    if (i > 1) return words[i];
                    if (d[i + 1][j] == d[i][j]) return words[i - 1];
                    return words[i];
                }
            }
        }
        return "";
    }
};

其实可以简化为一次遍历

// C++
class Solution {
public:
    string oddString(vector<string>& words) {
        int n = words.size(), bits = words[0].size();
        int diff = 0; // 只需要存一个差值即可
        // 依次看每一位
        for (int i = 1; i < bits; i++) {
            for (int j = 0; j < n; j++) {
                int t = words[j][i] - words[j][i - 1];
                if (j > 0 && t != diff) {
                    // 第一次出现, 当前这个单词, 与上一个单词, 在同一位置上的差值不一样
                    // 若第一次出现时j已经>=2, 那么至少0和1都是一样的, 那么不一样的一定是j本身
                    if (j >= 2) return words[j];
                    // 否则是j=1, 要看一下0和1究竟是谁
                    if (words[j + 1][i] - words[j + 1][i - 1] == t) return words[j - 1];
                    return words[j];
                }
                diff = t;
            }
        }
        return "";
    }
};

还可以求出某个字符串的diff数组,然后将diff作为哈希表的key,进行统计计数。最后求得计数为1的那个diff即可。

// C++
class Solution {
public:
    string diff(string& w) {
        string s;
        for (int i = 1; i < w.size(); i++) s += w[i] - w[i - 1];
        return s;
    }
    string oddString(vector<string>& words) {
        unordered_map<string, int> freq;
        for (auto& s : words) freq[diff(s)]++;
        for (auto& s : words) {
            if (freq[diff(s)] == 1) return s;
        }
        return "";
    }
};

2452. 距离字典两次编辑以内的单词

给你两个字符串数组 queriesdictionary 。数组中所有单词都只包含小写英文字母,且长度都相同。

一次 编辑 中,你可以从 queries 中选择一个单词,将任意一个字母修改成任何其他字母。从 queries 中找到所有满足以下条件的字符串:不超过 两次编辑内,字符串与 dictionary 中某个字符串相同。

请你返回 queries 中的单词列表,这些单词距离 dictionary 中的单词 编辑次数 不超过 两次 。单词返回的顺序需要与 queries 中原本顺序相同。

提示:

  • 1 <= queries.length, dictionary.length <= 100
  • n == queries[i].length == dictionary[j].length
  • 1 <= n <= 100
  • 所有 queries[i]dictionary[j] 都只包含小写英文字母。

示例

输入:queries = ["word","note","ants","wood"], dictionary = ["wood","joke","moat"]
输出:["word","note","wood"]
解释:
- 将 "word" 中的 'r' 换成 'o' ,得到 dictionary 中的单词 "wood" 。
- 将 "note" 中的 'n' 换成 'j' 且将 't' 换成 'k' ,得到 "joke" 。
- "ants" 需要超过 2 次编辑才能得到 dictionary 中的单词。
- "wood" 不需要修改(0 次编辑),就得到 dictionary 中相同的单词。
所以我们返回 ["word","note","wood"] 。

思路

看了一下数据范围,这道题可以直接用暴力,遍历queries中的全部单词,依次与dictionary中的全部单词做比对,时间复杂度在10^6级别。

// C++
class Solution {
public:
    vector<string> twoEditWords(vector<string>& queries, vector<string>& dictionary) {
        vector<string> ans;
        for (auto& s : queries) {
            for (auto& d : dictionary) {
                int cnt = 0; //不同的字母数
                for (int i = 0; i < s.size(); i++) {
                    if (s[i] != d[i]) cnt++;
                }
                if (cnt <= 2) {
                    ans.push_back(s);
                    break;
                }
            }
        }
        return ans;
    }
};

2453. 摧毁一系列目标

给你一个下标从 0 开始的数组 nums ,它包含若干正整数,表示数轴上你需要摧毁的目标所在的位置。同时给你一个整数 space

你有一台机器可以摧毁目标。给机器 输入 nums[i] ,这台机器会摧毁所有位置在 nums[i] + c * space 的目标,其中 c 是任意非负整数。你想摧毁 nums尽可能多 的目标。

请你返回在摧毁数目最多的前提下,nums[i]最小值

提示:

  • 1 <= nums.length <= 10^5
  • 1 <= nums[i] <= 10^9
  • 1 <= space <= 10^9

示例

输入:nums = [3,7,8,1,1,5], space = 2
输出:1
解释:如果我们输入 nums[3] ,我们可以摧毁位于 1,3,5,7,9,... 这些位置的目标。
这种情况下, 我们总共可以摧毁 5 个目标(除了 nums[2])。
没有办法摧毁多于 5 个目标,所以我们返回 nums[3]

思路

这道题需要注意解读题意。直观的来说,选择某个元素nums[i]之后,能够删除所有与nums[i]的距离为space倍数的位置。

比如选择nums[i] = 1space = 2,则能删除这些位置:1,3,5,7,9,...

我们需要将题目中的条件nums[i] + c * space,进行一下变换。设nums[i] + c * space = x,则选定nums[i]后,能够删除的所有位置就是x,其中c是任意非负整数。观察这个式子,其实我们可以对式子的两边做一下模运算。分别模space。能够得到

nums[i] % space = x % space,这其实也就是说,nums[i]x其实是关于space同余的。

选定一个数nums[i]后,能够删除的所有数的位置,其除以space后的余数,是等于nums[i]除以space后的余数的。

那么我们可以对nums中全部的数都做一下mod space的计算,并对余数进行计数。摧毁数目最多,那么就是出现次数最多的那个余数。

求得余数r后,我们再遍历一次nums,找到mod space = r的最小的nums[i]即可。一共需要2次遍历,时间复杂度为 O(n)O(n)

第一版代码:

// C++
class Solution {
public:
    int destroyTargets(vector<int>& nums, int space) {
        unordered_map<int, int> freq;  // 余数的出现次数
        int r = 0, maxFreq = 0;
        for (auto& i : nums) {
            if (++freq[i % space] > maxFreq) {
                maxFreq = freq[i % space];
                r = i % space;
            }
        }
        // 求得出现次数最大的余数后, 直接找出对应最小的nums[i]
        int ans = 1e9;
        for (auto& i : nums) {
            if (i % space == r && i < ans) ans = i;
        }
        return ans;
    }
};

提交后会发现WA,因为有些特殊情况没有考虑到,比如:

  • 当全部余数出现的频率都是1时,则需要直接返回nums的最小值
  • 可能有多个余数具有相同的最大出现次数,我们需要取nums[i]更小的那个余数(包含了上面的情况)
// C++
class Solution {
public:
    int destroyTargets(vector<int>& nums, int space) {
        unordered_map<int, int> freq; // 某个余数出现的次数
        unordered_map<int, int> min_v; // 某个余数对应的最小的nums[i]
        int r = 0, maxFreq = 0;
        for (auto& i : nums) {
            int x = i % space;
            // 不存在时
            if (min_v.find(x) == min_v.end()) min_v[x] = i;
            else min_v[x] = min(min_v[x], i);

            freq[x]++;
            if (freq[x] > maxFreq) {
                maxFreq = freq[x];
                r = x;
            } else if (freq[x] == maxFreq && min_v[x] < min_v[r]) {
                r = x;
            }
        }
        return min_v[r];
    }
};

后记:听y总讲解后,发现题目中要求c是非负数,这也就意味着,如果space = 2,则nums[i] = 1,能删除1, 3, 5, ...

nums[3],则能删除3, 5, 7, ...,只能删除往后的。而132的结果都是1,我们实际统计同余的次数,其实都是针对最小的nums[i]而言有效的。

2454. 下一个更大元素 IV

给你一个下标从 0 开始的非负整数数组 nums 。对于 nums 中每一个整数,你必须找到对应元素的 第二大 整数。

如果 nums[j] 满足以下条件,那么我们称它为 nums[i]第二大 整数:

  • j > i
  • nums[j] > nums[i]
  • 恰好存在 一个 k 满足 i < k < jnums[k] > nums[i]

如果不存在 nums[j] ,那么第二大整数为 -1

  • 比方说,数组 [1, 2, 4, 3] 中,1 的第二大整数是 42 的第二大整数是 334 的第二大整数是 -1

请你返回一个整数数组 answer ,其中 answer[i]nums[i] 的第二大整数。

提示:

  • 1 <= nums.length <= 10^5
  • 0 <= nums[i] <= 10^9

示例

输入:nums = [2,4,0,9,6]
输出:[9,6,6,-1,-1]
解释:
下标为 0 处:2 的右边,4 是大于 2 的第一个整数,9 是第二个大于 2 的整数。
下标为 1 处:4 的右边,9 是大于 4 的第一个整数,6 是第二个大于 4 的整数。
下标为 2 处:0 的右边,9 是大于 0 的第一个整数,6 是第二个大于 0 的整数。
下标为 3 处:右边不存在大于 9 的整数,所以第二大整数为 -1 。
下标为 4 处:右边不存在大于 6 的整数,所以第二大整数为 -1 。
所以我们返回 [9,6,6,-1,-1]

思路

审完题后,发现这道题求解的是:某个数右侧第二个比其大的数。而使用单调栈能求解出某个数右侧第一个比其大的数。那么我们用两次单调栈行不行呢?直接用两次单调栈貌似不行,当出现形如[2 6 4]这样的,6是第一个比2大的,4是第二个比2大的,但是4是大于6的。

换一种思路,我们使用一次单调栈,能够求出第一个比其大的数的位置(假设为i),那么我们第二次使用单调栈时,能不能加入一个条件呢?

  • 求解位置大于i的,且第一个比其大的数

好像想不太通。下面直接看题解。

单调栈+小根堆

从左往右进行遍历,使用单调栈求出右侧存在第一个比起大的数,然后将其放到小根堆中,继续往右遍历的过程中,如果出现了某个数大于小根堆的堆顶元素,则说明该数是堆顶元素的第二大的数。

通俗的说,我们使用单调栈,从左往右遍历时,对某个数x,当第一次遇到其右侧有比其大的数时,将x塞到小根堆中。小根堆中的数,都是当前已经找到其右侧第一个比其大的数,那么继续往右遍历时,当遍历到的数大于小根堆的堆顶,则找到第二大。

// C++
typedef pair<int, int> PII;
class Solution {
public:
    vector<int> secondGreaterElement(vector<int>& nums) {
        int n = nums.size();
        stack<int> stk;
        vector<int> ans(n, -1);
        priority_queue<PII, vector<PII>, greater<PII>> heap; // 小根堆
        for (int i = 0; i < n; i++) {
            // printf("i = %d\n", i);
            while (!heap.empty() && nums[i] > heap.top().first) {
                // 堆顶对应的元素, 找到其右侧第二大的
                ans[heap.top().second] = nums[i];
                heap.pop();
            }

            while (!stk.empty() && nums[stk.top()] < nums[i]) {
                // 对于nums[stk.top()], 其右侧存在第一个比起大的数
                // 将其插入到小根堆
                heap.push({nums[stk.top()], stk.top()});
                stk.pop();
            }
            stk.push(i);
        }
        return ans;
    }
};

再上一下y总的解法,挺牛的。

排序+维护下标

由于是求某个数右侧第二大的数(涉及到数的大小,已经下标)。那么我们把所有数,按照从大到小的顺序,依次枚举,而要求右侧,所以我们需要维护下标。

假设有一根线段,线段上出现的点,表示当前已经纳入考虑的数的下标

那么,我们从大到小,依次把数对应的下标放到这跟线段上。那么当枚举到某个数x上时,此时线段上放的都是大于该数的数的下标。我们只需要查找第二个大于数x下标的位置即可。由于可能存在相等的数,我们一次性处理所有值相同的数。否则,如果每次只处理一个数,当下一次再处理相同的数时,就不能满足线段上的数都是大于当前数这一语义了。

注:这里再次运用了,使用下标来对数据进行排序的这一技巧!!!

时间复杂度 O(nlogn)O(nlogn)

// C++
class Solution {
public:
    vector<int> secondGreaterElement(vector<int>& nums) {
        int n = nums.size();
        // 把下标存起来
        vector<int> p(n);
        for (int i = 0; i < n; i++) p[i] = i;
        // 从大到小排序, 只排下标, 不改变原数组
        sort(p.begin(), p.end(), [&](int a, int b){
            return nums[a] > nums[b];
        });
        // C++中的set是自带排序的
        
        vector<int> ans(n, -1);

        set<int> s;
        s.insert(n);
        s.insert(n + 1);// 保证一定存在大于某个数的第二个位置
        for (int i = 0; i < n; i++) {
            // 找到一段连续相等的数, 一起处理
            int j = i + 1;
            while (j < n && nums[p[j]] == nums[p[i]]) j++;
            // 已经插入到set中的下标, 对应的数, 都是比当前数大的
            // 因为set中存的是下标, 直接从set中找到第一个比当前下标大的, 然后将迭代器+1, 就是第二大数的下标
            for (int k = i; k < j; k++) {
                auto it = s.upper_bound(p[k]);
                ++it;
                if (*it < n) ans[p[k]] = nums[*it]; // 如果*it >= n,则说明不存在
            }
            // 将本次处理的数的下标全部插入到set中
            for (int k = i; k < j; k++) {
                s.insert(p[k]);
            }
            i = j - 1; // 下一个要处理的位置, 其实是j, 但是i会自动+1, 所以更新i=j-1
        }
        return ans;
    }
};

双单调栈

其实和单调栈+小根堆的思路差不多,开两个栈s1s2,其中s1中存的是当前还没有找到其右侧第一个比它大的那些数。而s2中存的是当前已经找到了其右侧第一个比它大的那些数。我们每次只要判断当前的数nums[i]是否比s2中的数大,即可。所以s2的栈顶必须是最小的。所以s2中也要保持元素的递减(栈顶最小)。故需要一个辅助栈。

// C++
class Solution {
public:
    vector<int> secondGreaterElement(vector<int>& nums) {
        int n = nums.size();
        stack<int> s1, s2;
        vector<int> ans(n, -1);
        for (int i = 0; i < n; i++) {
            while (s2.size() && nums[s2.top()] < nums[i]) {
                ans[s2.top()] = nums[i];
                s2.pop();
            }
            stack<int> tmp;
            while (s1.size() && nums[s1.top()] < nums[i]) {
                // 找到了第一个大的数
                tmp.push(s1.top());
                s1.pop();
            }
            // s1保持递减
            s1.push(i);
            while (tmp.size()) {
                s2.push(tmp.top());
                tmp.pop();
            }
        }
        return ans;
    }
};

总结

本场比赛很拉跨。只做出两题。

最近准备换租,当天晚上去楼上新的房子里打扫了卫生,有点累,做题的时候有点心不在焉,边做边和朋友聊天。哈哈哈,结果第一题花了45分钟才做出来。第三题也是在临近12点比赛结束时才发现规律, 等到提交通过时已经是12点4分了。

T1可以模拟,也可以用哈希表;

T2暴力;

T3是数学问题,需要察觉到规律就是模运算,同余;

T4是单调栈的变形运用,注意,如果扩展一下,求右侧第k大,那么y总的那种下标排序+有序列表是比较有效的。另外,这里再次看到了,根据数组元素从大到小,对数组的下标进行排序,这样的处理技巧。