Skip to content

Latest commit

 

History

History
4743 lines (4378 loc) · 131 KB

LeetCode高频题.md

File metadata and controls

4743 lines (4378 loc) · 131 KB

LeetCode 高频题

哈希

1. 两数之和

class Solution {
public:
    vector<int> twoSum(vector<int>& nums, int target) {
        unordered_map<int, int> table;
        for (int i = 0; i < nums.size(); i++) {
            int l = target - nums[i];
            if (table.find(l) != table.end()) {
                return {table[l], i};
            }
            table[nums[i]] = i;
        }
        return {};
    }
};

49. 字母异位词分组

给你一个字符串数组,请你将 字母异位词 组合在一起。可以按任意顺序返回结果列表。
字母异位词 是由重新排列源单词的所有字母得到的一个新单词。
输入: strs = ["eat", "tea", "tan", "ate", "nat", "bat"]
输出: [["bat"],["nat","tan"],["ate","eat","tea"]]

class Solution {
public:
    vector<vector<string>> groupAnagrams(vector<string>& strs) {
        unordered_map<string, vector<string>> table;
        for (const auto &str : strs) {
            string key = str;
            sort(key.begin(), key.end());
            table[key].emplace_back(str);
        }
        vector<vector<string>> ans;
        for (const auto &item : table) {
            ans.emplace_back(item.second);
        }
        return ans;
    }
};

128. 最长连续序列

给定一个未排序的整数数组 nums ,找出数字连续的最长序列(不要求序列元素在原数组中连续)的长度。
请你设计并实现时间复杂度为 O(n) 的算法解决此问题。

class Solution {
public:
    int longestConsecutive(vector<int>& nums) {
        unordered_set<int> table(nums.begin(), nums.end());
        int ans = 0;
        for (const auto &num : nums) {
            if (table.find(num - 1) == table.end()) {
                int end = num;
                while (table.find(end + 1) != table.end()) {
                    end++;
                }
                ans = max(ans, end - num + 1);
            }
        }
        return ans;
    }
};

双指针

283. 移动零

给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。
请注意 ,必须在不复制数组的情况下原地对数组进行操作。

class Solution {
public:
    void moveZeroes(vector<int>& nums) {
        int k = 0;
        for (const auto &num : nums) {
            if (num != 0) {
                nums[k++] = num;
            }
        }
        for (int i = k; i < nums.size(); i++) {
            nums[i] = 0;
        }
    }
};

11. 盛最多水的容器

给定一个长度为 n 的整数数组 height 。有 n 条垂线,第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。
找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。

class Solution {
public:
    int maxArea(vector<int>& height) {
        int left = 0, right = height.size() - 1;
        int leftMax = 0, rightMax = 0;
        int ans = 0;
        while (left < right) {
            leftMax = max(leftMax, height[left]);
            rightMax = max(rightMax, height[right]);
            if (leftMax < rightMax) {
                ans = max(ans, leftMax * (right - left));
                left++;
            } else {
                ans = max(ans, rightMax * (right - left));
                right--;
            }
        }
        return ans;
    }
};

15. 三数之和

class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        vector<vector<int>> ans;
        sort(nums.begin(), nums.end());
        for (int i = 0; i < nums.size(); i++) {
            if (i && nums[i] == nums[i - 1]) {
                continue;
            }
            for (int j = i + 1, k = nums.size() - 1; j < k; j++) {
                if (j > i + 1 && nums[j] == nums[j - 1]) {
                    continue;
                }
                while (j < k - 1 && nums[i] + nums[j] + nums[k - 1] >= 0) {
                    k--;
                }
                if (nums[i] + nums[j] + nums[k] == 0) {
                    ans.push_back({nums[i], nums[j], nums[k]});
                }
            }
        }
        return ans;
    }
};

209. 长度最小的子数组

给定一个含有 n 个正整数的数组和一个正整数 target。 找出该数组中满足其总和大于等于 target 的长度最小的子数组,并返回其长度。如果不存在符合条件的子数组,返回 0。

class Solution {
public:
    int minSubArrayLen(int target, vector<int>& nums) {
        int ans = INT_MAX;
        int sum = 0;
        for (int i = 0, j = 0; i < nums.size(); i++) {
            sum += nums[i];
            while (sum - nums[j] >= target) {
                sum -= nums[j++];
            }
            if (sum >= target) {
                ans = min(ans, i - j + 1);
            }
        }
        return ans == INT_MAX ? 0 : ans;
    }
};

42. 接雨水

给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。

class Solution {
public:
    int trap(vector<int>& height) {
        int left = 0, right = height.size() - 1;
        int leftMax = 0, rightMax = 0;
        int ans = 0;
        while (left < right) {
            leftMax = max(leftMax, height[left]);
            rightMax = max(rightMax, height[right]);
            if (leftMax < rightMax) {
                ans += leftMax - height[left];
                left++;
            } else {
                ans += rightMax - height[right];
                right--;
            }
        }
        return ans;
    }
};

581. 最短无序连续子数组

给你一个整数数组 nums ,你需要找出一个 连续子数组 ,如果对这个子数组进行升序排序,那么整个数组都会变为升序排序。

// 输入:nums = [2,6,4,8,10,9,15]
// 输出:5
// 解释:你只需要对 [6, 4, 8, 10, 9] 进行升序排序,那么整个表都会变为升序排序。
class Solution {
public:
    int findUnsortedSubarray(vector<int>& nums) {
        int left = 0, right = nums.size() - 1;
        while (left + 1 < nums.size() && nums[left] <= nums[left + 1]) {
            left++;
        }
        if (left == right) {
            return 0;
        }
        while (right - 1 >= 0 && nums[right - 1] <= nums[right]) {
            right--;
        }

        for (int i = left + 1; i < nums.size(); i++) {
            while (left >= 0 && nums[left] > nums[i]) {
                left--;
            }
        }
        for (int i = right - 1; i >= 0; i--) {
            while (right < nums.size() && nums[right] < nums[i]) {
                right++;
            }
        }
        return right - left - 1;
    }
};

滑动窗口

3. 无重复字符的最长子串

class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        unordered_map<char, int> sCount;
        int ans = 0;
        for (int i = 0, j = 0; i < s.size(); i++) {
            sCount[s[i]]++;
            while (sCount[s[i]] > 1) {
                sCount[s[j++]]--;
            }
            ans = max(ans, i - j + 1);
        }
        return ans;
    }
};

438. 找到字符串中所有字母异位词

给定两个字符串 s 和 p,找到 s 中所有 p 的 异位词 的子串,返回这些子串的起始索引。不考虑答案输出的顺序。
异位词 指由相同字母重排列形成的字符串(包括相同的字符串)。
输入: s = "cbaebabacd", p = "abc"
输出: [0,6]

class Solution {
public:
    vector<int> findAnagrams(string s, string p) {
        vector<int> ans;
        if (s.size() < p.size()) {
            return ans;
        }
        vector<int> sCount(26);
        vector<int> pCount(26);
        for (int i = 0; i < p.size(); i++) {
            sCount[s[i] - 'a']++;
            pCount[p[i] - 'a']++;
        }
        if (sCount == pCount) {
            ans.emplace_back(0);
        }
        for (int i = 1; i < s.size() - p.size() + 1; i++) {
            sCount[s[i - 1] - 'a']--;
            sCount[s[i + p.size() - 1] - 'a']++;
            if (sCount == pCount) {
                ans.emplace_back(i);
            }
        }
        return ans;
    }
};

子串

560. 和为 K 的子数组

输入:nums = [1,1,1], k = 2
输出:2

class Solution {
public:
    int subarraySum(vector<int>& nums, int k) {
        vector<int> sum(nums.size());
        sum[0] = nums[0];
        for (int i = 1; i < sum.size(); i++) {
            sum[i] = sum[i - 1] + nums[i];
        }
        unordered_map<int, int> table;
        table[0] = 1;
        int ans = 0;
        for (int i = 0; i < nums.size(); i++) {
            ans += table[sum[i] - k];
            table[sum[i]]++;
        }
        return ans;
    }
};

239. 滑动窗口最大值

输入:nums = [1,3,-1,-3,5,3,6,7], k = 3
输出:[3,3,5,5,6,7]

class Solution {
public:
    vector<int> maxSlidingWindow(vector<int>& nums, int k) {
        vector<int> ans;
        deque<int> q;
        for (int i = 0; i < nums.size(); i++) {
            if (q.size() && q.front() < i - k + 1) {
                q.pop_front();
            }
            while (q.size() && nums[i] >= nums[q.back()]) {
                q.pop_back();
            }
            q.push_back(i);
            if (i >= k - 1) {
                ans.emplace_back(nums[q.front()]);
            }
        }
        return ans;
    }
};

76. 最小覆盖子串

输入:s = "ADOBECODEBANC", t = "ABC" 输出:"BANC"

class Solution {
public:
    string minWindow(string s, string t) {
        unordered_map<char, int> sCount, tCount;
        for (const auto &ch : t) {
            tCount[ch]++;
        }
        int cnt = 0;
        string ans;
        for (int i = 0, j = 0; i < s.size(); i++) {
            sCount[s[i]]++;
            if (sCount[s[i]] <= tCount[s[i]]) {
                cnt++;
            }
            while (sCount[s[j]] > tCount[s[j]]) {
                sCount[s[j++]]--;
            }
            if (cnt == t.size()) {
                if (ans.empty() || i - j + 1 < ans.size()) {
                    ans = s.substr(j, i - j + 1);
                }
            }
        }
        return ans;
    }
};

普通数组

912. 排序数组

// 堆排序
class Solution {
    void down(vector<int> &nums, int u, int len) {
        int t = u;
        if (2 * u + 1 < len && nums[2 * u + 1] < nums[t]) {
            t = 2 * u + 1;
        }
        if (2 * u + 2 < len && nums[2 * u + 2] < nums[t]) {
            t = 2 * u + 2;
        }
        if (t != u) {
            swap(nums[t], nums[u]);
            down(nums, t, len);
        }
    }

    void heap_sort(vector<int> &nums) {
        int len = nums.size();
        for (int i = len / 2 - 1; i >= 0; i--) {
            down(nums, i, len);
        }
        while (len--) {
            swap(nums[0], nums[len]);
            down(nums, 0, len);
        }
        reverse(nums.begin(), nums.end());
    }

public:
    vector<int> sortArray(vector<int>& nums) {
        heap_sort(nums);
        return nums;
    }
};

53. 最大子数组和

输入:nums = [-2,1,-3,4,-1,2,1,-5,4]
输出:6
解释:连续子数组 [4,-1,2,1] 的和最大,为 6。

class Solution {
public:
    int maxSubArray(vector<int>& nums) {
        int ans = INT_MIN;
        for (int i = 0, last = 0; i < nums.size(); i++) {
            last = nums[i] + max(last, 0);
            ans = max(ans, last);
        }
        return ans;
    }
};

56. 合并区间

输入:intervals = [[1,3],[2,6],[8,10],[15,18]]
输出:[[1,6],[8,10],[15,18]]

class Solution {
public:
    vector<vector<int>> merge(vector<vector<int>>& intervals) {
        vector<vector<int>> ans;
        if (intervals.empty()) {
            return ans;
        }
        sort(intervals.begin(), intervals.end());
        int left = intervals[0][0], right = intervals[0][1];
        for (int i = 1; i < intervals.size(); i++) {
            if (intervals[i][0] <= right) {
                right = max(right, intervals[i][1]);
            } else {
                ans.push_back({left, right});
                left = intervals[i][0];
                right = intervals[i][1];
            }
        }
        ans.push_back({left, right});
        return ans;
    }
};

253. 会议室 II

alias: https://www.lintcode.com/problem/919/solution/57831

// 输入: intervals = [(0,30),(5,10),(15,20)]
// 输出: 2
// 解释:
// 需要两个会议室
// 会议室1:(0,30)
// 会议室2:(5,10),(15,20)
class Solution {
public:
    /**
     * @param intervals: an array of meeting time intervals
     * @return: the minimum number of conference rooms required
     */
    int minMeetingRooms(vector<Interval> &intervals) {
        // Write your code here
        if (intervals.empty()) {
            return 0;
        }

        priority_queue<int, vector<int>, greater<int>> q;
        sort(intervals.begin(), intervals.end(), [](const Interval &left, const Interval &right) {
            return left.start < right.start || (left.start == right.start && left.end < right.end);
        });

        q.push({intervals[0].end});
        for (int i = 1; i < intervals.size(); i++) {
            if (intervals[i].start >= q.top()) {
                q.pop();
            }
            q.push(intervals[i].end);
        }
        return q.size();
    }
};

179. 最大数

给定一组非负整数 nums,重新排列每个数的顺序(每个数不可拆分)使之组成一个最大的整数。 输入:nums = [3,30,34,5,9] 输出:"9534330"

class Solution {
public:
    string largestNumber(vector<int>& nums) {
        sort(nums.begin(), nums.end(), [](const int &x, const int &y) {
            return to_string(x) + to_string(y) > to_string(y) + to_string(x);
        });
        if (nums[0] == 0) {
            return "0";
        }
        string ans;
        for (const auto &num : nums) {
            ans += to_string(num);
        }
        return ans;
    }
};

189. 轮转数组

输入: nums = [1,2,3,4,5,6,7], k = 3
输出: [5,6,7,1,2,3,4]

class Solution {
public:
    void rotate(vector<int>& nums, int k) {
        k = k % nums.size();
        reverse(nums.begin(), nums.end());
        reverse(nums.begin(), nums.begin() + k);
        reverse(nums.begin() + k, nums.end());
    }
};

238. 除自身以外数组的乘积

class Solution {
public:
    vector<int> productExceptSelf(vector<int>& nums) {
        vector<int> forward(nums.size(), 1);
        for (int i = 1; i < nums.size(); i++) {
            forward[i] = nums[i - 1] * forward[i - 1];
        }
        for (int i = nums.size() - 1, backward = 1; i >= 0; i--) {
            forward[i] = forward[i] * backward;
            backward = backward * nums[i];
        }
        return forward;
    }
};

41. 缺失的第一个正数

给你一个未排序的整数数组 nums ,请你找出其中没有出现的最小的正整数。
请你实现时间复杂度为 O(n) 并且只使用常数级别额外空间的解决方案。
输入:nums = [3,4,-1,1]
输出:2

class Solution {
public:
    int firstMissingPositive(vector<int>& nums) {
        int n = nums.size();
        for (int i = 0; i < n; i++) {
            while (nums[i] >= 1 && nums[i] <= n && nums[i] != nums[nums[i] - 1]) {
                swap(nums[i], nums[nums[i] - 1]);
            }
        }
        for (int i = 0; i < n; i++) {
            if (nums[i] != i + 1) {
                return i + 1;
            }
        }
        return n + 1;
    }
};

448. 找到所有数组中消失的数字

class Solution {
public:
    vector<int> findDisappearedNumbers(vector<int>& nums) {
        for (int i = 0; i < nums.size(); i++) {
            while (nums[i] != nums[nums[i] - 1]) {
                swap(nums[i], nums[nums[i] - 1]);
            }
        }
        vector<int> ans;
        for (int i = 0; i < nums.size(); i++) {
            if (nums[i] != i + 1) {
                ans.emplace_back(i + 1);
            }
        }
        return ans;
    }
};

406. 根据身高重建队列

输入:people = [[7,0],[4,4],[7,1],[5,0],[6,1],[5,2]] 输出:[[5,0],[7,0],[5,2],[6,1],[4,4],[7,1]]

class Solution {
public:
    vector<vector<int>> reconstructQueue(vector<vector<int>>& people) {
        sort(people.begin(), people.end(), [](const vector<int> &left, const vector<int> &right) {
            return left[0] < right[0] || (left[0] == right[0] && left[1] > right[1]);
        });
        vector<vector<int>> ans(people.size());
        for (const auto &person : people) {
            int spaces = person[1] + 1;
            for (int i = 0; i < people.size(); i++) {
                if (ans[i].empty()) {
                    --spaces;
                    if (!spaces) {
                        ans[i] = person;
                        break;
                    }
                }
            }
        }
        return ans;
    }
};

矩阵

73. 矩阵置零

给定一个 m x n 的矩阵,如果一个元素为 0 ,则将其所在行和列的所有元素都设为 0 。请使用 原地 算法。

class Solution {
public:
    void setZeroes(vector<vector<int>>& matrix) {
        int m = matrix.size(), n = matrix[0].size();
        bool row0 = false, col0 = false;

        for (int i = 0; i < m; i++) {
            if (!matrix[i][0]) {
                col0 = true;
            }
        }

        for (int j = 0; j < n; j++) {
            if (!matrix[0][j]) {
                row0 = true;
            }
        }

        for (int i = 1; i < m; i++) {
            for (int j = 0; j < n; j++) {
                if (!matrix[i][j]) {
                    matrix[i][0] = 0;
                }
            }
        }

        for (int j = 1; j < n; j++) {
            for (int i = 0; i < m; i++) {
                if (!matrix[i][j]) {
                    matrix[0][j] = 0;
                }
            }
        }

        for (int i = 1; i < m; i++) {
            for (int j = 0; j < n; j++) {
                if (!matrix[i][0]) {
                    matrix[i][j] = 0;
                }
            }
        }

        for (int j = 1; j < n; j++) {
            for (int i = 0; i < m; i++) {
                if (!matrix[0][j]) {
                    matrix[i][j] = 0;
                }
            }
        }

        for (int i = 0; i < m; i++) {
            if (col0) {
                matrix[i][0] = 0;
            }
        }

        for (int j = 0; j < n; j++) {
            if (row0) {
                matrix[0][j] = 0;
            }
        }
    }
};

54. 螺旋矩阵

给你一个 m 行 n 列的矩阵 matrix ,请按照 顺时针螺旋顺序 ,返回矩阵中的所有元素。

class Solution {
    vector<int> dx = {0, 1, 0, -1};
    vector<int> dy = {1, 0, -1, 0};
    vector<int> ans;
    vector<vector<bool>> visited;

public:
    vector<int> spiralOrder(vector<vector<int>>& matrix) {
        int m = matrix.size();
        int n = matrix[0].size();
        visited = vector<vector<bool>>(m, vector<bool>(n));
        for (int x = 0, y = 0, dir = 0, i = 0; i < m * n; i++) {
            ans.emplace_back(matrix[x][y]);
            visited[x][y] = true;
            int a = x + dx[dir];
            int b = y + dy[dir];
            if (a < 0 || a >= m || b < 0 || b >= n || visited[a][b]) {
                dir = (dir + 1) % dx.size();
                a = x + dx[dir];
                b = y + dy[dir];
            }
            x = a, y = b;
        }
        return ans;
    }
};

48. 旋转图像

给定一个 n × n 的二维矩阵 matrix 表示一个图像。请你将图像顺时针旋转 90 度。
你必须在 原地 旋转图像,这意味着你需要直接修改输入的二维矩阵。请不要 使用另一个矩阵来旋转图像。

class Solution {
public:
    void rotate(vector<vector<int>>& matrix) {
        for (int i = 0; i < matrix.size(); i++) {
            for (int j = 0; j < i; j++) {
                swap(matrix[i][j], matrix[j][i]);
            }
        }

        for (int i = 0; i < matrix.size(); i++) {
            for (int j = 0; j < matrix.size() / 2; j++) {
                swap(matrix[i][j], matrix[i][matrix.size() - j - 1]);
            }
        }
    }
};

字符串

8. 字符串转换整数 (atoi)

请你来实现一个 myAtoi(string s) 函数,使其能将字符串转换成一个 32 位有符号整数。 函数 myAtoi(string s) 的算法如下:

  • 空格:读入字符串并丢弃无用的前导空格(" ")
  • 符号:检查下一个字符(假设还未到字符末尾)为 '-' 还是 '+'。如果两者都不存在,则假定结果为正。
  • 转换:通过跳过前置零来读取该整数,直到遇到非数字字符或到达字符串的结尾。如果没有读取数字,则结果为0。
  • 舍入:如果整数数超过 32 位有符号整数范围 [−2^31, 2^31 − 1],需要截断这个整数,使其保持在这个范围内。具体来说,小于 −2^31 的整数应该被舍入为 −2^31 ,大于 2^31 − 1 的整数应该被舍入为 2^31 − 1。
class Solution {
public:
    int myAtoi(string s) {
        int k = 0;
        while (k < s.size() && s[k] == ' ') {
            k++;
        }
        if (k == s.size()) {
            return 0;
        }

        int minus = 1;
        if (s[k] == '-') {
            minus = -1;
            k++;
        } else if (s[k] == '+') {
            k++;
        }

        long long ans = 0;
        while (k < s.size() && s[k] >= '0' && s[k] <= '9') {
            ans = ans * 10 + s[k] - '0';
            k++;
            if (ans > INT_MAX) {
                break;
            }
        }
        ans *= minus;
        if (ans > INT_MAX) {
            ans = INT_MAX;
        }
        if (ans < INT_MIN) {
            ans = INT_MIN;
        }
        return ans;
    }
};

14. 最长公共前缀

class Solution {
public:
    string longestCommonPrefix(vector<string>& strs) {
        for (int i = 0; i < strs[0].size(); i++) {
            for (int j = 1; j < strs.size(); j++) {
                if (i == strs[j].size() || strs[0][i] != strs[j][i]) {
                    return strs[0].substr(0, i);
                }
            }
        }
        return strs[0];
    }
};

151. 翻转字符串里的单词

class Solution {
public:
    string reverseWords(string s) {
        reverse(s.begin(), s.end());
        int idx = 0;
        for (int i = 0; i < s.size(); i++) {
            if (s[i] == ' ') {
                continue;
            }
            if (idx != 0) {
                s[idx++] = ' ';
            }
            int j = i;
            while (j < s.size() && s[j] != ' ') {
                s[idx++] = s[j++];
            }
            reverse(s.begin() + idx - (j - i), s.begin() + idx);
            i = j;
        }
        s.erase(s.begin() + idx, s.end());
        return s;
    }
};

165. 比较版本号

输入:version1 = "1.2", version2 = "1.10" 输出:-1

class Solution {
public:
    int compareVersion(string version1, string version2) {
        for (int start1 = 0, start2 = 0; start1 < version1.size() || start2 < version2.size(); ) {
            int end1 = start1, end2 = start2;
            while (end1 < version1.size() && version1[end1] != '.') {
                end1++;
            }
            while (end2 < version2.size() && version2[end2] != '.') {
                end2++;
            }
            int v1 = end1 == start1 ? 0 : stoi(version1.substr(start1, end1 - start1));
            int v2 = end2 == start2 ? 0 : stoi(version2.substr(start2, end2 - start2));
            if (v1 < v2) {
                return -1;
            }
            if (v1 > v2) {
                return 1;
            }
            start1 = end1 + 1;
            start2 = end2 + 1;
        }
        return 0;
    }
};

415. 字符串相加

输入:num1 = "11", num2 = "123" 输出:"134"

class Solution {
public:
    string addStrings(string num1, string num2) {
        string ans;
        int i = num1.size() - 1;
        int j = num2.size() - 1;
        int c = 0;
        while (i >= 0 || j >= 0 || c) {
            if (i >= 0) {
                c += num1[i] - '0';
                i--;
            }
            if (j >= 0) {
                c += num2[j] - '0';
                j--;
            }
            ans.push_back(c % 10 + '0');
            c /= 10;
        }
        reverse(ans.begin(), ans.end());
        return ans;
    }
};

43. 字符串相乘

class Solution {
public:
    string multiply(string num1, string num2) {
        vector<int> a;
        vector<int> b;
        reverse(num1.begin(), num1.end());
        reverse(num2.begin(), num2.end());
        for (const auto &ch : num1) {
            a.emplace_back(ch - '0');
        }
        for (const auto &ch : num2) {
            b.emplace_back(ch - '0');
        }

        vector<int> c(a.size() + b.size());
        for (int i = 0; i < a.size(); i++) {
            for (int j = 0; j < b.size(); j++) {
                c[i + j] += a[i] * b[j];
            }
        }
        for (int i = 0, t = 0; i < c.size(); i++) {
            t += c[i];
            c[i] = t % 10;
            t /= 10;
        }

        int k = c.size() - 1;
        while (k > 0 && c[k] == 0) {
            k--;
        }

        string ans;
        while (k >= 0) {
            ans += c[k] + '0';
            k--;
        }
        return ans;
    }
};

224. 基本计算器
227. 基本计算器 II
772. 基本计算器 III

我们最终要实现的计算器功能如下:

  1. 输入一个字符串,可以包含+ - * /、数字、括号以及空格,你的算法返回运算结果。
  2. 要符合运算法则,括号的优先级最高,先乘除后加减。
  3. 除号是整数除法,无论正负都向 0 取整(5/2=2,-5/2=-2)。
  4. 可以假定输入的算式一定合法,且计算过程不会出现整型溢出,不会出现除数为 0 的意外情况。
class Solution {
    int dfs(const string &s, int &i) {
        stack<int> stk;
        int num = 0;
        char sign = '+';
        for (; i < s.size(); i++) {
            // 当前字符是空格, 如果不是字符最后的位置, 直接跳过
            if (s[i] == ' ' && i != s.size() - 1) {
                continue;
            }
            // 当前字符是数字
            else if (isdigit(s[i])) {
                num = 10 * num + (s[i] - '0');
            }
            // 当前字符是 '('
            else if (s[i] == '(') {
                i++;
                num = dfs(s, i);
                i++;
            }

            // 第一种情况, 遇到 '+' , '-' , '*' , '/', ')' 需要进行运算
            // 第二种情况, 遇到字符串的尾部 (尾部可能是一个数字, 可能是 ')' , 也可能是' ', 所以这两种情况之间有重叠)
            if (!isdigit(s[i]) || i == s.size() - 1) {
                int pre;   
                if (sign == '+') {
                    stk.push(num);
                } else if (sign == '-') {
                    stk.push(-num);
                } else if (sign == '*') {
                    pre = stk.top();
                    stk.pop();
                    stk.push(pre * num);
                } else if (sign == '/') {
                    pre = stk.top();
                    stk.pop();
                    stk.push(pre / num);
                }

                // 只有递归过程才会遇到 ')', 上面运算完了需要额外进行 break
                if (s[i] == ')') {
                    break;
                }

                sign = s[i];
                num = 0;
            }
        }

        // 计算栈中所有元素的和
        int ans = 0;
        while (stk.size()) {
            ans += stk.top();
            stk.pop();
        }
        return ans;
    }

public:
    int calculate(string s) {
        int i = 0;
        return dfs(s, i);
    }
};

301. 删除无效的括号

给你一个由若干括号和字母组成的字符串 s ,删除最小数量的无效括号,使得输入的字符串有效。 返回所有可能的结果。答案可以按 任意顺序 返回。

class Solution {
    vector<string> ans;

    bool isValid(const string &s) {
        int cnt = 0;
        for (int i = 0; i < s.size(); i++) {
            if (s[i] == '(') {
                cnt++;
            } else if (s[i] == ')') {
                cnt--;
                if (cnt < 0) {
                    return false;
                }
            }
        }
        return cnt == 0;
    }

    void dfs(string s, int cur, int lremove, int rremove) {
        if (lremove == 0 && rremove == 0) {
            if (isValid(s)) {
                ans.emplace_back(s);
            }
            return ;
        }
        for (int i = cur; i < s.size(); i++) {
            // 去重
            if (i != cur && s[i] == s[i - 1]) {
                continue;
            }
            // 如果剩余的字符无法满足去掉的数量要求,直接返回
            if (lremove + rremove > s.size() - i) {
                return ;
            }
            // 尝试去掉一个左括号
            if (lremove > 0 && s[i] == '(') {
                dfs(s.substr(0, i) + s.substr(i + 1), i, lremove - 1, rremove);
            }
            // 尝试去掉一个右括号
            if (rremove > 0 && s[i] == ')') {
                dfs(s.substr(0, i) + s.substr(i + 1), i, lremove, rremove)
            }
        }
    }

public:
    vector<string> removeInvalidParentheses(string s) {
        int lremove = 0;
        int rremove = 0;

        for (const auto &ch : s) {
            if (ch == '(') {
                lremove++;
            } else if (ch == ')') {
                if (lremove == 0) {
                    rremove++;
                } else {
                    lremove--;
                }
            }
        }
        dfs(s, 0, lremove, rremove);
        return ans;
    }
};

链表

160. 相交链表

class Solution {
public:
    ListNode *getIntersectionNode(ListNode *headA, ListNode *headB) {
        ListNode *node1 = headA;
        ListNode *node2 = headB;
        while (node1 != node2) {
            node1 = node1 ? node1->next : headB;
            node2 = node2 ? node2->next : headA;
        }
        return node1;
    }
};

206. 反转链表

class Solution {
public:
    // 迭代
    ListNode* reverseList(ListNode* head) {
        ListNode *prev = nullptr;
        ListNode *node = head;
        while (node) {
            ListNode *next = node->next;
            node->next = prev;
            prev = node;
            node = next;
        }
        return prev;
    }
    // 递归
    ListNode* reverseList(ListNode* head) {
        if (head == nullptr || head->next == nullptr) {
            return head;
        }
        ListNode *node = reverseList(head->next);
        head->next->next = head;
        head->next = nullptr;
        return node;
    }
};

92. 反转链表 II

给你单链表的头指针 head 和两个整数 left 和 right ,其中 left <= right 。请你反转从位置 left 到位置 right 的链表节点,返回 反转后的链表 。

class Solution {
public:
    ListNode* reverseBetween(ListNode* head, int left, int right) {
        ListNode *dummpy = new ListNode(0, head);
        ListNode *a = head;
        ListNode *b = head;
        ListNode *prev = dummpy;
        while (--left) {
            prev = prev->next;
            a = a->next;
        }
        while (--right) {
            b = b->next;
        }
        ListNode *bnext = b->next;
        b->next = nullptr;
        reverseList(a);

        prev->next = b;
        a->next = bnext;
        return dummpy->next;
    }
};

83. 删除排序链表中的重复元素

给定一个已排序的链表的头 head , 删除所有重复的元素,使每个元素只出现一次 。返回 已排序的链表 。

class Solution {
public:
    ListNode* deleteDuplicates(ListNode* head) {
        if (head == nullptr) {
            return head;
        }

        ListNode *dummpy = new ListNode(0, head);
        ListNode *prev = dummpy;
        ListNode *node = head;
        while (node->next) {
            if (node->val != node->next->val) {
                prev->next = node;
                prev = prev->next;
            }
            node = node->next;
        }
        prev->next = node;
        return dummpy->next;
    }
};

82. 删除排序链表中的重复元素 II

给定一个已排序的链表的头 head , 删除原始链表中所有重复数字的节点,只留下不同的数字 。返回 已排序的链表 。

class Solution {
public:
    ListNode* deleteDuplicates(ListNode* head) {
        if (head == nullptr) {
            return nullptr;
        }

        ListNode *dummpy = new ListNode(0, head);
        ListNode *prev = dummpy;
        ListNode *node = head;
        bool deleted = false;
        while (node->next) {
            if (node->val == node->next->val) {
                deleted = true;
            } else {
                if (deleted) {
                    prev->next = node->next;
                    deleted = false;
                } else {
                    prev->next = node;
                    prev = prev->next;
                }
            }
            node = node->next;
        }
        if (deleted) {
            prev->next = nullptr;
        }
        return dummpy->next;
    }
};

129. 求根到叶子节点数字之和

计算从根节点到叶节点生成的所有数字之和。

class Solution {
    int ans = 0;

    void dfs(TreeNode *node, int path) {
        if (node == nullptr) {
            return ;
        }

        path = path * 10 + node->val;
        if (node->left == nullptr && node->right == nullptr) {
            ans += path;
            return ;
        }
        
        dfs(node->left, path);
        dfs(node->right, path);
    }

public:
    int sumNumbers(TreeNode* root) {
        dfs(root, 0);
        return ans;
    }
};

234. 回文链表

用 O(n) 时间复杂度和 O(1) 空间复杂度解决此题

class Solution {
public:
    bool isPalindrome(ListNode* head) {
        int num = 0;
        for (auto p = head; p; p = p->next) {
            num++;
        }
        if (num <= 1) {
            return true;
        }
        int half = num / 2;
        ListNode *a = head;
        while (half--) {
            a = a->next;
        }
        a = reverseList(a);
        ListNode *b = head;
        while (a && b) {
            if (a->val != b->val) {
                return false;
            }
            a = a->next;
            b = b->next;
        }
        return true;
    }
};

141. 环形链表

class Solution {
public:
    bool hasCycle(ListNode *head) {
        if (head == nullptr || head->next == nullptr) {
            return false;
        }
        ListNode *fast = head;
        ListNode *slow = head;
        while (fast && fast->next) {
            slow = slow->next;
            fast = fast->next->next;
            if (slow == fast) {
                return true;
            }
        }
        return false;
    }
};

142. 环形链表 II

class Solution {
public:
    ListNode *detectCycle(ListNode *head) {
        if (head == nullptr || head->next == nullptr) {
            return nullptr;
        }
        ListNode *slow = head;
        ListNode *fast = head;
        while (fast && fast->next) {
            slow = slow->next;
            fast = fast->next->next;
            if (slow == fast) {
                break;
            }
        }
        if (fast == nullptr || fast->next == nullptr) {
            return nullptr;
        }

        slow = head;
        while (slow != fast) {
            slow = slow->next;
            fast = fast->next;
        }
        return slow;
    }
};

143. 重排链表

给定一个单链表 L 的头节点 head ,单链表 L 表示为: L0 → L1 → … → Ln-1 → Ln 请将其重新排列后变为: L0 → Ln → L1 → Ln-1 → L2 → Ln-2 → … 不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。

class Solution {
public:
    void reorderList(ListNode* head) {
        int n = 0;
        for (ListNode *p = head; p; p = p->next) {
            n++;
        }

        ListNode *ahead = head;
        n = (n + 1) / 2;
        ListNode *node = head;
        while (--n) {
            node = node->next;
        }
        ListNode *bhead = node->next;
        node->next = nullptr;

        bhead = reverseList(bhead);
        ListNode *dummy = new ListNode(0);
        ListNode *prev = dummy;
        while (ahead && bhead) {
            prev->next = ahead;
            ahead = ahead->next;
            prev = prev->next;

            prev->next = bhead;
            bhead = bhead->next;
            prev = prev->next;
        }
        if (ahead) {
            prev->next = ahead;
        }
    }
};

21. 合并两个有序链表

class Solution {
public:
    ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {
        ListNode *dummpy = new ListNode(0);
        ListNode *prev = dummpy;
        while (list1 && list2) {
            if (list1->val <= list2->val) {
                prev->next = list1;
                prev = prev->next;
                list1 = list1->next;
            } else {
                prev->next = list2;
                prev = prev->next;
                list2 = list2->next;
            }
        }
        if (list1) {
            prev->next = list1;
        }
        if (list2) {
            prev->next = list2;
        }
        return dummpy->next;
    }
};

2. 两数相加

输入:l1 = [2,4,3], l2 = [5,6,4]
输出:[7,0,8]
解释:342 + 465 = 807.

class Solution {
public:
    ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) {
        ListNode *dummpy = new ListNode(0);
        ListNode *prev = dummpy;
        int c = 0;
        while (l1 || l2 || c) {
            if (l1) {
                c += l1->val;
                l1 = l1->next;
            }
            if (l2) {
                c += l2->val;
                l2 = l2->next;
            }
            prev->next = new ListNode(c % 10);
            prev = prev->next;
            c /= 10;
        }
        return dummpy->next;
    }
};

19. 删除链表的倒数第 N 个结点

输入:head = [1,2,3,4,5], n = 2
输出:[1,2,3,5]

class Solution {
public:
    ListNode* removeNthFromEnd(ListNode* head, int n) {
        ListNode *dummpy = new ListNode(0, head);
        ListNode *low = head;
        ListNode *fast = head;

        for (int i = 0; i < n && fast; i++) {
            fast = fast->next;
        }

        ListNode *prev = dummpy;

        while (fast) {
            low = low->next;
            fast = fast->next;
            prev = prev->next;
        }

        prev->next = low->next;
        return dummpy->next;
    }
};

24. 两两交换链表中的节点

class Solution {
public:
    ListNode* swapPairs(ListNode* head) {
        ListNode *dummpy = new ListNode(0, head);
        ListNode *prev = dummpy;
        while (prev->next && prev->next->next) {
            ListNode *a = prev->next;
            ListNode *b = prev->next->next;
            prev->next = b;
            a->next = b->next;
            b->next = a;

            prev = a;
        }
        return dummpy->next;
    }
};

25. K 个一组翻转链表

输入:head = [1,2,3,4,5], k = 3 输出:[3,2,1,4,5]

class Solution {
    ListNode *getK(ListNode *node, int k) {
        for (int i = 0; i < k - 1 && node; i++) {
            node = node->next;
        }
        return node;
    }

public:
    ListNode* reverseKGroup(ListNode* head, int k) {
        ListNode *dummpy = new ListNode(0, head);
        ListNode *prev = dummpy;
        ListNode *tail = getK(head, k);
        while (tail) {
            ListNode *nextHead = tail->next;
            tail->next = nullptr;
            reverseList(head);
            prev->next = tail;
            head->next = nextHead;

            prev = head;
            head = nextHead;
            tail = getK(head, k);
        }
        return dummpy->next;
    }
};

138. 随机链表的复制

class Solution {
    unordered_map<Node*, Node*> table;

public:
    Node* copyRandomList(Node* head) {
        if (head == nullptr) {
            return nullptr;
        }

        if (table.find(head) == table.end()) {
            Node *newHead = new Node(head->val);
            table[head] = newHead;
            newHead->next = copyRandomList(head->next);
            newHead->random = copyRandomList(head->random);
        }
        return table[head];
    }
};

148. 排序链表

在 O(n log n) 时间复杂度和常数级空间复杂度下,对链表进行排序

class Solution {
    ListNode *merge(ListNode *head1, ListNode *head2) {
        ListNode *dummpy = new ListNode(0);
        ListNode *pre = dummpy;
        ListNode *node1 = head1;
        ListNode *node2 = head2;
        while (node1 && node2) {
            if (node1->val <= node2->val) {
                pre->next = node1;
                pre = pre->next;
                node1 = node1->next;
            } else {
                pre->next = node2;
                pre = pre->next;
                node2 = node2->next;
            }
        }
        if (node1) {
            pre->next = node1;
        }
        if (node2) {
            pre->next = node2;
        }
        return dummpy->next;
    }

public:
    ListNode* sortList(ListNode* head) {
        int length = 0;
        for (ListNode *p = head; p; p = p->next) {
            length++;
        }
        ListNode *dummpy = new ListNode(0, head);
        for (int subLength = 1; subLength < length; subLength += subLength) {
            ListNode *pre = dummpy;
            ListNode *cur = dummpy->next;
            while (cur) {
                ListNode *head1 = cur;
                for (int i = 1; i < subLength && cur->next; i++) {
                    cur = cur->next;
                }
                ListNode *head2 = cur->next;
                cur->next = nullptr;
                cur = head2;
                for (int i = 1; i < subLength && cur && cur->next; i++) {
                    cur = cur->next;
                }
                ListNode *next = nullptr;
                if (cur) {
                    next = cur->next;
                    cur->next = nullptr;
                }
                ListNode *merged = merge(head1, head2);
                pre->next = merged;
                while (pre->next) {
                    pre = pre->next;
                }
                cur = next;
            }
        }
        return dummpy->next;
    }
};

23. 合并 K 个升序链表

class Solution {
    ListNode *merge(ListNode *head1, ListNode *head2) {
        ListNode *dummpy = new ListNode(0);
        ListNode *pre = dummpy;
        ListNode *node1 = head1;
        ListNode *node2 = head2;
        while (node1 && node2) {
            if (node1->val <= node2->val) {
                pre->next = node1;
                pre = pre->next;
                node1 = node1->next;
            } else {
                pre->next = node2;
                pre = pre->next;
                node2 = node2->next;
            }
        }
        if (node1) {
            pre->next = node1;
        }
        if (node2) {
            pre->next = node2;
        }
        return dummpy->next;
    }

    ListNode *mergeKListsHelper(vector<ListNode*> &lists, int left, int right) {
        if (left > right) {
            return nullptr;
        }
        if (left == right) {
            return lists[left];
        }
        int mid = left + right >> 1;
        return merge(mergeKListsHelper(lists, left, mid), mergeKListsHelper(lists, mid + 1, right));
    }

public:
    ListNode* mergeKLists(vector<ListNode*>& lists) {
        return mergeKListsHelper(lists, 0, lists.size() - 1);
    }
};

146. LRU 缓存

class LRUCache {
    class Node {
    public:
        int key, value;
        Node *prev, *next;
        Node(int key, int value) : key(key), value(value), prev(nullptr), next(nullptr) {}
    };
    Node *head, *tail;
    unordered_map<int, Node*> table;
    int n;

    void remove(Node *node) {
        node->prev->next = node->next;
        node->next->prev = node->prev;
    }

    void insert(Node *node) {
        node->prev = tail->prev;
        node->next = tail;
        node->prev->next = node;
        tail->prev = node;
    }

public:
    LRUCache(int capacity) {
        n = capacity;
        head = new Node(0, 0);
        tail = new Node(0, 0);
        head->next = tail;
        tail->prev = head;
    }
    
    int get(int key) {
        if (table.find(key) == table.end()) {
            return -1;
        }
        Node *node = table[key];
        remove(node);
        insert(node);
        return node->value;
    }
    
    void put(int key, int value) {
        if (table.find(key) == table.end()) {
            if (n == table.size()) {
                Node *first = head->next;
                remove(first);
                table.erase(first->key);
            }
            Node *node = new Node(key, value);
            insert(node);
            table[key] = node;
        } else {
            Node *node = table[key];
            node->value = value;
            remove(node);
            insert(node);
        }
    }
};

二叉树

94. 二叉树的中序遍历

class Solution {
    vector<int> ans;

public:
    vector<int> inorderTraversal(TreeNode* root) {
        if (root == nullptr) {
            return ans;
        }
        inorderTraversal(root->left);
        ans.emplace_back(root->val);
        inorderTraversal(root->right);
        return ans;
    }
};

104. 二叉树的最大深度

class Solution {
public:
    int maxDepth(TreeNode* root) {
        if (root == nullptr) {
            return 0;
        }
        int left = maxDepth(root->left);
        int right = maxDepth(root->right);
        return max(left, right) + 1;
    }
};

226. 翻转二叉树

class Solution {
public:
    TreeNode* invertTree(TreeNode* root) {
        if (root == nullptr) {
            return root;
        }
        TreeNode *left = invertTree(root->left);
        TreeNode *right = invertTree(root->right);
        root->left = right;
        root->right = left;
        return root;
    }
};

101. 对称二叉树

class Solution {
    bool isSymmetricHelper(TreeNode *left, TreeNode *right) {
        if (!left && !right) {
            return true;
        }
        if (!left || !right) {
            return false;
        }
        if (left->val != right->val) {
            return false;
        }
        return isSymmetricHelper(left->left, right->right) && isSymmetricHelper(left->right, right->left);
    }

public:
    bool isSymmetric(TreeNode* root) {
        return isSymmetricHelper(root, root);
    }
};

543. 二叉树的直径

二叉树的 直径 是指树中任意两个节点之间最长路径的 长度 。这条路径可能经过也可能不经过根节点 root 。

class Solution {
    int ans;

    int depth(TreeNode *node) {
        if (node == nullptr) {
            return 0;
        }
        int left = depth(node->left);
        int right = depth(node->right);
        ans = max(ans, left + right);
        return max(left, right) + 1;
    }

public:
    int diameterOfBinaryTree(TreeNode* root) {
        ans = 0;
        depth(root);
        return ans;
    }
};

102. 二叉树的层序遍历

class Solution {
public:
    vector<vector<int>> levelOrder(TreeNode* root) {
        vector<vector<int>> ans;
        if (root == nullptr) {
            return ans;
        }
        queue<TreeNode*> q;
        q.push(root);
        while (q.size()) {
            int sz = q.size();
            vector<int> path;
            while (sz--) {
                TreeNode *front = q.front();
                q.pop();
                path.emplace_back(front->val);
                if (front->left) {
                    q.push(front->left);
                }
                if (front->right) {
                    q.push(front->right);
                }
            }
            ans.emplace_back(path);
        }
        return ans;
    }
};

103. 二叉树的锯齿形层序遍历

给定一个二叉树,返回其节点值的锯齿形层序遍历。 即先从左往右,再从右往左进行下一层遍历,以此类推,层与层之间交替进行。

class Solution {
public:
    vector<vector<int>> zigzagLevelOrder(TreeNode* root) {
        vector<vector<int>> ans;
        if (root == nullptr) {
            return ans;
        }
        queue<TreeNode*> q;
        q.push(root);
        int level = 0;
        while (q.size()) {
            level++;
            int sz = q.size();
            vector<int> path;
            while (sz--) {
                TreeNode *front = q.front();
                q.pop();
                path.emplace_back(front->val);
                if (front->left) {
                    q.push(front->left);
                }
                if (front->right) {
                    q.push(front->right);
                }
            }
            if (level % 2 == 0) {
                reverse(path.begin(), path.end());
            }
            ans.emplace_back(path);
        }
        return ans;
    }
};

108. 将有序数组转换为二叉搜索树

class Solution {
    TreeNode *buildTree(vector<int> &nums, int left, int right) {
        if (left > right) {
            return nullptr;
        }
        int mid = left + right >> 1;
        TreeNode *node = new TreeNode(nums[mid]);
        node->left = buildTree(nums, left, mid - 1);
        node->right = buildTree(nums, mid + 1, right);
        return node;
    }

public:
    TreeNode* sortedArrayToBST(vector<int>& nums) {
        return buildTree(nums, 0, nums.size() - 1);
    }
};

110. 平衡二叉树

给定一个二叉树,判断它是否是 平衡二叉树(是指该树所有节点的左右子树的深度相差不超过 1。)

class Solution {
    int depth(TreeNode *node) {
        if (node == nullptr) {
            return 0;
        }
        int left = depth(node->left);
        int right = depth(node->right);
        return max(left, right) + 1;
    }

    bool dfs(TreeNode *node) {
        if (node == nullptr) {
            return true;
        }
        int left = depth(node->left);
        int right = depth(node->right);
        if (abs(right - left) > 1) {
            return false;
        }
        return dfs(node->left) && dfs(node->right);
    }

public:
    bool isBalanced(TreeNode* root) {
        return dfs(root);
    }
};

98. 验证二叉搜索树

class Solution {
    using LL = long long;

    bool isValidBSTHelper(TreeNode *node, LL lower, LL upper) {
        if (node == nullptr) {
            return true;
        }

        if (node->val <= lower || node->val >= upper) {
            return false;
        }

        return isValidBSTHelper(node->left, lower, node->val) && isValidBSTHelper(node->right, node->val, upper);
    }

public:
    bool isValidBST(TreeNode* root) {
        return isValidBSTHelper(root, LONG_LONG_MIN, LONG_LONG_MAX);
    }
};

230. 二叉搜索树中第 K 小的元素

给定一个二叉搜索树的根节点 root ,和一个整数 k ,请你设计一个算法查找其中第 k 小的元素(从 1 开始计数)。

class Solution {
    int countNum(TreeNode *node) {
        if (node == nullptr) {
            return 0;
        }
        int left = countNum(node->left);
        int right = countNum(node->right);
        return left + right + 1;
    }

public:
    int kthSmallest(TreeNode* root, int k) {
        int leftNum = countNum(root->left);
        if (leftNum >= k) {
            return kthSmallest(root->left, k);
        } else if (leftNum + 1 == k) {
            return root->val;
        } else {
            return kthSmallest(root->right, k - leftNum - 1);
        }
    }
};

199. 二叉树的右视图

class Solution {
    vector<int> ans;

    void levelOrder(TreeNode *node, int level) {
        if (node == nullptr) {
            return ;
        }

        if (level >= ans.size()) {
            ans.emplace_back(node->val);
        }
        levelOrder(node->right, level + 1);
        levelOrder(node->left, level + 1);
    }

public:
    vector<int> rightSideView(TreeNode* root) {
        levelOrder(root, 0);
        return ans;
    }
};

114. 二叉树展开为链表

给你二叉树的根结点 root ,请你将它展开为一个单链表:

  • 展开后的单链表应该同样使用 TreeNode ,其中 right 子指针指向链表中下一个结点,而左子指针始终为 null 。
  • 展开后的单链表应该与二叉树 先序遍历 顺序相同。
class Solution {
public:
    void flatten(TreeNode* root) {
        while (root) {
            TreeNode *node = root->left;
            if (node) {
                while (node->right) {
                    node = node->right;
                }
                node->right = root->right;
                root->right = root->left;
                root->left = nullptr;
            }
            root = root->right;
        }
    }
};

105. 从前序与中序遍历序列构造二叉树

class Solution {
    unordered_map<int, int> table;

    TreeNode *build(vector<int> &preorder, vector<int> &inorder, int pl, int pr, int il, int ir) {
        if (pl > pr) {
            return nullptr;
        }
        TreeNode *node = new TreeNode(preorder[pl]);
        int index = table[preorder[pl]];
        node->left = build(preorder, inorder, pl + 1, pl + 1 + index - 1 - il + 1 - 1, il, index - 1);
        node->right = build(preorder, inorder, pl + 1 + index - 1 - il + 1 - 1 + 1, pr, index + 1, ir);
        return node;
    }

public:
    TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder) {
        for (int i = 0; i < inorder.size(); i++) {
            table[inorder[i]] = i;
        }
        return build(preorder, inorder, 0, preorder.size() - 1, 0, inorder.size() - 1);
    }
};

112. 路径总和

给你二叉树的根节点 root 和一个表示目标和的整数 targetSum 。判断该树中是否存在 根节点到叶子节点 的路径,这条路径上所有节点值相加等于目标和 targetSum 。如果存在,返回 true ;否则,返回 false 。

class Solution {
    bool dfs(TreeNode *node, const int &targetSum, int sum) {
        if (node == nullptr) {
            return false;
        }
        sum += node->val;
        if (node->left == nullptr && node->right == nullptr) {
            return sum == targetSum;
        }
        return dfs(node->left, targetSum, sum) || dfs(node->right, targetSum, sum);
    }

public:
    bool hasPathSum(TreeNode* root, int targetSum) {
        if (root == nullptr) {
            return false;
        }

        return dfs(root, targetSum, 0);
    }
};

113. 路径总和 II

给你二叉树的根节点 root 和一个整数目标和 targetSum ,找出所有 从根节点到叶子节点 路径总和等于给定目标和的路径。

class Solution {
    vector<vector<int>> ans;
    vector<int> path;

    void dfs(TreeNode *node, const int &targetSum, int sum) {
        if (node == nullptr) {
            return ;
        }
        sum += node->val;
        path.emplace_back(node->val);
        if (node->left == nullptr && node->right == nullptr && sum == targetSum) {
            ans.emplace_back(path);
        }
        dfs(node->left, targetSum, sum);
        dfs(node->right, targetSum, sum);
        path.pop_back();
    }
    
public:
    vector<vector<int>> pathSum(TreeNode* root, int targetSum) {
        dfs(root, targetSum, 0);
        return ans;
    }
};

437. 路径总和 III

给定一个二叉树的根节点 root ,和一个整数 targetSum ,求该二叉树里节点值之和等于 targetSum 的 路径 的数目。
路径 不需要从根节点开始,也不需要在叶子节点结束,但是路径方向必须是向下的(只能从父节点到子节点)。

class Solution {
    using LL = long long;
    unordered_map<LL, LL> table;
    LL ans;

    void dfs(TreeNode *node, LL targetSum, LL sum) {
        if (node == nullptr) {
            return ;
        }
        sum += node->val;
        ans += table[sum - targetSum];
        table[sum]++;
        dfs(node->left, targetSum, sum);
        dfs(node->right, targetSum, sum);
        table[sum]--;
    }

public:
    int pathSum(TreeNode* root, int targetSum) {
        ans = 0;
        table[0] = 1;
        dfs(root, targetSum, 0);
        return ans;
    }
};

236. 二叉树的最近公共祖先

class Solution {
public:
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
        if (root == nullptr) {
            return nullptr;
        }
        if (root == p || root == q) {
            return root;
        }
        TreeNode *left = lowestCommonAncestor(root->left, p, q);
        TreeNode *right = lowestCommonAncestor(root->right, p, q);
        if (left && right) {
            return root;
        }
        return left ? left : right;
    }
};

124. 二叉树中的最大路径和

输入:root = [-10,9,20,null,null,15,7]
输出:42
解释:最优路径是 15 -> 20 -> 7 ,路径和为 15 + 20 + 7 = 42

class Solution {
    int ans;

    int dfs(TreeNode *node) {
        if (node == nullptr) {
            return 0;
        }
        int left = max(0, dfs(node->left));
        int right = max(0, dfs(node->right));
        ans = max(ans, left + right + node->val);
        return node->val + max(left, right);
    }

public:
    int maxPathSum(TreeNode* root) {
        ans = INT_MIN;
        dfs(root);
        return ans;
    }
};

572. 另一棵树的子树

class Solution {
    vector<int> s{0};
    vector<int> p{0};

    void preOrder(TreeNode *node, vector<int> &vec) {
        if (node == nullptr) {
            vec.emplace_back(-1e4 - 1);
            return ;
        }
        vec.emplace_back(node->val);
        preOrder(node->left, vec);
        preOrder(node->right, vec);
    }

public:
    bool isSubtree(TreeNode* root, TreeNode* subRoot) {
        preOrder(root, s);
        preOrder(subRoot, p);
        int ans = -1;
        int n = s.size() - 1, m = p.size() - 1;
        vector<int> next(m + 1, 0);
        for (int i = 2, j = 0; i <= m; i++) {
            while (j && p[i] != p[j + 1]) {
                j = next[j];
            }
            if (p[i] == p[j + 1]) {
                j++;
            }
            next[i] = j;
        }
        for (int i = 1, j = 0; i <= n; i++) {
            while (j && s[i] != p[j + 1]) {
                j = next[j];
            }
            if (s[i] == p[j + 1]) {
                j++;
            }
            if (j == m) {
                return true;
            }
        }
        return false;
    }
};

538. 把二叉搜索树转换为累加树

class Solution {
    int sum = 0;

    void dfs(TreeNode *node) {
        if (node == nullptr) {
            return ;
        }
        dfs(node->right);
        sum += node->val;
        node->val = sum;
        dfs(node->left);
    }

public:
    TreeNode* convertBST(TreeNode* root) {
        dfs(root);
        return root;
    }
};

617. 合并二叉树

class Solution {
public:
    TreeNode* mergeTrees(TreeNode* root1, TreeNode* root2) {
        if (root1 == nullptr) {
            return root2;
        }
        if (root2 == nullptr) {
            return root1;
        }

        root1->val += root2->val;
        
        TreeNode *left = mergeTrees(root1->left, root2->left);
        TreeNode *right = mergeTrees(root1->right, root2->right);
        root1->left = left;
        root1->right = right;
        return root1;
    }
};

662. 二叉树最大宽度

class Solution {
    using ULL = unsigned long long;
    unordered_map<int, ULL> levelMin;
    ULL ans;

    void dfs(TreeNode *node, ULL index, int level) {
        if (node == nullptr) {
            return ;
        }
        if (levelMin.find(level) == levelMin.end()) {
            levelMin[level] = index;
        }
        ans = max(ans, index - levelMin[level] + 1);
        dfs(node->left, index * 2, level + 1);
        dfs(node->right, index * 2 + 1, level + 1);
    }

public:
    int widthOfBinaryTree(TreeNode* root) {
        dfs(root, 1, 0);
        return ans;
    }
};

297. 二叉树的序列化与反序列化

class Codec {
public:
    // Encodes a tree to a single string.
    string serialize(TreeNode* root) {
        if (root == nullptr) {
            return "None";
        }
        string s;
        s = to_string(root->val) + ",";
        s += serialize(root->left);
        s += "," + serialize(root->right);
        return s;
    }

    TreeNode *deserialize_helper(vector<string> &data, int &pos) {
        if (data[pos] == "None") {
            pos += 1;
            return nullptr;
        }
        int val = stoi(data[pos]);
        TreeNode *node = new TreeNode(val);
        pos += 1;
        node->left = deserialize_helper(data, pos);
        node->right = deserialize_helper(data, pos);
        return node;
    }

    // Decodes your encoded data to tree.
    TreeNode* deserialize(string data) {
        vector<string> data_vec;
        string item = "";
        for (auto &ch : data) {
            if (ch == ',') {
                data_vec.emplace_back(item);
                item = "";
            } else {
                item += ch;
            }
        }
        data_vec.emplace_back(item);
        int pos = 0;
        return deserialize_helper(data_vec, pos);
    }
};

3249. 统计好节点的数目

如果一个节点的所有子节点为根的子树包含的节点数相同,则认为该节点是一个好节点。
输入:edges = [[0,1],[0,2],[1,3],[1,4],[2,5],[2,6]]
输出:7

class Solution {
    int ans = 0;
    vector<vector<int>> graph;

    int dfs(int cur, int prev) {
        int son_size = 0;
        for (const auto &son : graph[cur]) {
            if (son != prev) {
                son_size++;
            }
        }
        if (son_size == 0) {
            ans++;
            return 1;
        }

        int son_sum = 0;
        int sum = 0;
        bool check = true;
        for (const auto &son : graph[cur]) {
            if (son == prev) {
                continue;
            }
            int new_son_sum = dfs(son, cur);
            if (son_sum == 0) {
                son_sum = new_son_sum;
            }
            if (son_sum != new_son_sum) {
                check = false;
            }
            sum += new_son_sum;
        }
        sum++;

        if (check) {
            ans++;
        }
        return sum;
    }

public:
    int countGoodNodes(vector<vector<int>>& edges) {
        int n = edges.size() + 1;
        graph.resize(n);
        for (const auto &e : edges) {
            int a = e[0], b = e[1];
            graph[a].push_back(b);
            graph[b].push_back(a);
        }

        dfs(0, -1);
        return ans;
    }
};

图论

200. 岛屿数量

给你一个由 '1'(陆地)和 '0'(水)组成的的二维网格,请你计算网格中岛屿的数量。

class Solution {
    vector<int> dx{-1, 0, 1, 0};
    vector<int> dy{0, 1, 0, -1};

    void dfs(vector<vector<char>> &grid, int x, int y) {
        grid[x][y] = '0';
        for (int i = 0; i < dx.size(); i++) {
            int a = x + dx[i];
            int b = y + dy[i];
            if (a < 0 || a >= grid.size() || b < 0 || b >= grid[0].size() || grid[a][b] == '0') {
                continue;
            }
            dfs(grid, a, b);
        }
    }

public:
    int numIslands(vector<vector<char>>& grid) {
        int ans = 0;
        for (int i = 0; i < grid.size(); i++) {
            for (int j = 0; j < grid[0].size(); j++) {
                if (grid[i][j] == '1') {
                    ans++;
                    dfs(grid, i, j);
                }
            }
        }
        return ans;
    }
};

695. 岛屿的最大面积

class Solution {
    vector<int> dx{-1, 0, 1, 0};
    vector<int> dy{0, 1, 0, -1};
    int ans = 0;

    void dfs(vector<vector<int>> &grid, int x, int y, int &cnt) {
        grid[x][y] = 0;
        cnt++;
        ans = max(ans, cnt);
        for (int i = 0; i < dx.size(); i++) {
            int a = x + dx[i];
            int b = y + dy[i];
            if (a < 0 || a >= grid.size() || b < 0 || b >= grid[0].size() || !grid[a][b]) {
                continue;
            }
            dfs(grid, a, b, cnt);
        }
    }

public:
    int maxAreaOfIsland(vector<vector<int>>& grid) {
        for (int i = 0; i < grid.size(); i++) {
            for (int j = 0; j < grid[0].size(); j++) {
                if (grid[i][j] == 1) {
                    int cnt = 0;
                    dfs(grid, i, j, cnt);
                }
            }
        }
        return ans;
    }
};

994. 腐烂的橘子

在给定的网格中,每个单元格可以有以下三个值之一:

  • 值 0 代表空单元格;
  • 值 1 代表新鲜橘子;
  • 值 2 代表腐烂的橘子。

每分钟,任何与腐烂橘子(在 4 个正方向上)相邻的新鲜橘子都会腐烂。
返回直到单元格中没有新鲜橘子为止所必须经过的最小分钟数。如果不可能,返回 -1。

class Solution {
    using PII = pair<int, int>;
    vector<int> dx = {-1, 0, 1, 0};
    vector<int> dy = {0, 1, 0, -1};

public:
    int orangesRotting(vector<vector<int>>& grid) {
        queue<PII> q;
        for (int i = 0; i < grid.size(); i++) {
            for (int j = 0; j < grid[0].size(); j++) {
                if (grid[i][j] == 2) {
                    q.push({i, j});
                }
            }
        }
        int ans = 0;
        if (q.size()) {
            ans--;
        }
        while (q.size()) {
            ans++;
            int sz = q.size();
            while (sz--) {
                auto front = q.front();
                q.pop();
                int x = front.first;
                int y = front.second;
                for (int i = 0; i < dx.size(); i++) {
                    int a = x + dx[i];
                    int b = y + dy[i];
                    if (a < 0 || a >= grid.size() || b < 0 || b >= grid[0].size() || grid[a][b] != 1) {
                        continue;
                    }
                    grid[a][b] = 2;
                    q.push({a, b});
                }
            }
        }
        for (int i = 0; i < grid.size(); i++) {
            for (int j = 0; j < grid[0].size(); j++) {
                if (grid[i][j] == 1) {
                    return -1;
                }
            }
        }
        return ans;
    }
};

207. 课程表

你这个学期必须选修 numCourses 门课程,记为 0 到 numCourses - 1 。
在选修某些课程之前需要一些先修课程。 先修课程按数组 prerequisites 给出,其中 prerequisites[i] = [ai, bi] ,表示如果要学习课程 ai 则 必须 先学习课程 bi 。

  • 例如,先修课程对 [0, 1] 表示:想要学习课程 0 ,你需要先完成课程 1 。

请你判断是否可能完成所有课程的学习?如果可以,返回 true ;否则,返回 false 。

class Solution {
public:
    bool canFinish(int numCourses, vector<vector<int>>& prerequisites) {
        vector<vector<int>> edge(numCourses);
        vector<int> indegree(numCourses);
        for (const auto &e : prerequisites) {
            int a = e[0], b = e[1];
            edge[b].emplace_back(a);
            indegree[a]++;
        }
        queue<int> q;
        for (int i = 0; i < indegree.size(); i++) {
            if (indegree[i] == 0) {
                q.push(i);
            } 
        }
        int cnt = 0;
        while (q.size()) {
            cnt++;
            int front = q.front();
            q.pop();
            for (const auto &node : edge[front]) {
                indegree[node]--;
                if (indegree[node] == 0) {
                    q.push(node);
                }
            }
        }
        return cnt == numCourses;
    }
};

208. 实现 Trie (前缀树)

// Trie trie = new Trie();
// trie.insert("apple");
// trie.search("apple");   // 返回 True
// trie.search("app");     // 返回 False
// trie.startsWith("app"); // 返回 True
// trie.insert("app");
// trie.search("app");     // 返回 True
class Trie {
    class Node {
    public:
        bool isEnd;
        vector<Node*> son;
        Node() : isEnd(false), son(26, nullptr) {}
    };
    Node *root;

public:
    Trie() {
        root = new Node();
    }
    
    void insert(string word) {
        Node *node = root;
        for (const auto &ch : word) {
            int index = ch - 'a';
            if (!node->son[index]) {
                node->son[index] = new Node();
            }
            node = node->son[index];
        }
        node->isEnd = true;
    }
    
    bool search(string word) {
        Node *node = root;
        for (const auto &ch : word) {
            int index = ch - 'a';
            if (!node->son[index]) {
                return false;
            }
            node = node->son[index];
        }
        return node->isEnd;
    }
    
    bool startsWith(string prefix) {
        Node *node = root;
        for (const auto &ch : prefix) {
            int index = ch - 'a';
            if (!node->son[index]) {
                return false;
            }
            node = node->son[index];
        }
        return true;
    }
};

399. 除法求值

// 输入:equations = [["a","b"],["b","c"]], values = [2.0,3.0], queries = [["a","c"],["b","a"],["a","e"],["a","a"],["x","x"]]
// 输出:[6.00000,0.50000,-1.00000,1.00000,-1.00000]
// 解释:
// 条件:a / b = 2.0, b / c = 3.0
// 问题:a / c = ?, b / a = ?, a / e = ?, a / a = ?, x / x = ?
// 结果:[6.0, 0.5, -1.0, 1.0, -1.0 ]
// 注意:x 是未定义的 => -1.0
class Solution {
public:
    vector<double> calcEquation(vector<vector<string>>& equations, vector<double>& values, vector<vector<string>>& queries) {
        unordered_map<string, unordered_map<string, double>> graph;
        for (int i = 0; i < equations.size(); i++) {
            string a = equations[i][0];
            string b = equations[i][1];
            graph[a][b] = values[i];
            graph[b][a] = 1.0 / values[i];
        }

        vector<double> ans(queries.size(), -1.0);
        for (int i = 0; i < queries.size(); i++) {
            string c = queries[i][0];
            string d = queries[i][1];
            if (!graph.count(c) || !graph.count(d)) {
                continue;
            }
            queue<pair<string, double>> q;
            unordered_map<string, bool> visited;
            q.push({c, 1.0});
            visited[c] = true;
            while (!q.empty()) {
                auto node = q.front();
                q.pop();
                if (node.first == d) {
                    ans[i] = node.second;
                    break;
                }
                for (const auto &next : graph[node.first]) {
                    if (!visited[next.first]) {
                        visited[next.first] = true;
                        q.push({next.first, node.second * next.second});
                    }
                }
            }
        }
        return ans;
    }
};

回溯

46. 全排列

class Solution {
    vector<vector<int>> ans;
    vector<int> path;
    vector<bool> used;

    void dfs(vector<int> &nums, int cur) {
        if (cur == nums.size()) {
            ans.emplace_back(path);
            return ;
        }

        for (int i = 0; i < nums.size(); i++) {
            if (!used[i]) {
                used[i] = true;
                path.emplace_back(nums[i]);
                dfs(nums, cur + 1);
                path.pop_back();
                used[i] = false;
            }
        }
    }

public:
    vector<vector<int>> permute(vector<int>& nums) {
        used = vector<bool>(nums.size());
        dfs(nums, 0);
        return ans;
    }
};

78. 子集

给你一个整数数组 nums ,数组中的元素 互不相同 。返回该数组所有可能的子集(幂集)。
解集 不能 包含重复的子集。你可以按 任意顺序 返回解集。
输入:nums = [1,2,3]
输出:[[],[1],[2],[1,2],[3],[1,3],[2,3],[1,2,3]]

class Solution {
    vector<vector<int>> ans;
    vector<int> path;

    void dfs(vector<int> &nums, int cur) {
        if (cur == nums.size()) {
            ans.emplace_back(path);
            return ;
        }
        path.emplace_back(nums[cur]);
        dfs(nums, cur + 1);
        path.pop_back();
        dfs(nums, cur + 1);
    }

public:
    vector<vector<int>> subsets(vector<int>& nums) {
        dfs(nums, 0);
        return ans;
    }
};

17. 电话号码的字母组合

给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。答案可以按 任意顺序 返回。

class Solution {
    unordered_map<char, string> table = {
        {'2', "abc"},
        {'3', "def"},
        {'4', "ghi"},
        {'5', "jkl"},
        {'6', "mno"},
        {'7', "pqrs"},
        {'8', "tuv"},
        {'9', "wxyz"},
    };
    vector<string> ans;

    void dfs(const string &digits, int cur, string path) {
        if (cur == digits.size()) {
            ans.emplace_back(path);
            return;
        }
        for (const auto &ch : table[digits[cur]]) {
            dfs(digits, cur + 1, path + ch);
        }
    }

public:
    vector<string> letterCombinations(string digits) {
        if (digits == "") {
            return ans;
        }
        dfs(digits, 0, "");
        return ans;
    }
};

39. 组合总和

输入:candidates = [2,3,6,7], target = 7
输出:[[2,2,3],[7]]
这是完全背包问题,但是要求输出所有解,所以需要使用回溯算法

class Solution {
    vector<vector<int>> ans;
    vector<int> path;

    void dfs(vector<int> &candidates, int cur, int target) {
        if (target < 0) {
            return ;
        }
        if (target == 0) {
            ans.emplace_back(path);
            return;
        }
        if (cur == candidates.size()) {
            return;
        }
        path.emplace_back(candidates[cur]);
        dfs(candidates, cur, target - candidates[cur]);
        path.pop_back();
        dfs(candidates, cur + 1, target);
    }

public:
    vector<vector<int>> combinationSum(vector<int>& candidates, int target) {
        dfs(candidates, 0, target);
        return ans;
    }
};

22. 括号生成

输入:n = 3
输出:["((()))","(()())","(())()","()(())","()()()"]

class Solution {
    vector<string> ans;

    void dfs(int n, int left, int right, string path) {
        if (left == n && right == n) {
            ans.emplace_back(path);
            return;
        }
        if (left < n) {
            dfs(n, left + 1, right, path + "(");
        }
        if (right < n && left > right) {
            dfs(n, left, right + 1, path + ")");
        }
    }

public:
    vector<string> generateParenthesis(int n) {
        dfs(n, 0, 0, "");
        return ans;
    }
};

79. 单词搜索

给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中,返回 true ;否则,返回 false 。

class Solution {
    vector<int> dx = {-1, 0, 1, 0};
    vector<int> dy = {0, 1, 0, -1};

    bool dfs(vector<vector<char>> &board, const string &word, int cur, int x, int y) {
        if (board[x][y] != word[cur]) {
            return false;
        }
        if (cur == word.size() - 1) {
            return true;
        }
        char tmp = board[x][y];
        board[x][y] = '.';
        for (int i = 0; i < dx.size(); i++) {
            int a = x + dx[i];
            int b = y + dy[i];
            if (a < 0 || a >= board.size() || b < 0 || b >= board[0].size() || board[a][b] == '.') {
                continue;
            }
            if (dfs(board, word, cur + 1, a, b)) {
                return true;
            }
        }
        board[x][y] = tmp;
        return false;
    }
    
public:
    bool exist(vector<vector<char>>& board, string word) {
        for (int i = 0; i < board.size(); i++) {
            for (int j = 0; j < board[0].size(); j++) {
                if (dfs(board, word, 0, i, j)) {
                    return true;
                }
            }
        }
        return false;
    }
};

131. 分割回文串

输入:s = "aab"
输出:[["a","a","b"],["aa","b"]]

class Solution {
    vector<vector<bool>> f;
    vector<vector<string>> ans;
    vector<string> path;

    void dfs(const string &s, int cur) {
        if (cur == s.size()) {
            ans.emplace_back(path);
            return ;
        }
        for (int i = cur; i < s.size(); i++) {
            if (f[cur][i]) {
                path.emplace_back(s.substr(cur, i - cur + 1));
                dfs(s, i + 1);
                path.pop_back();
            }
        }
    }

public:
    vector<vector<string>> partition(string s) {
        int n = s.size();
        f = vector<vector<bool>>(n, vector<bool>(n));

        for (int j = 0; j < n; j++) {
            for (int i = 0; i <= j; i++) {
                if (s[i] != s[j]) {
                    continue;
                }
                if (i == j || i + 1 == j) {
                    f[i][j] = true;
                } else {
                    f[i][j] = f[i + 1][j - 1];
                }
            }
        }
        dfs(s, 0);
        return ans;
    }
};

51. N 皇后

class Solution {
    int n_;
    vector<bool> col, dg, udg;
    vector<vector<string>> ans;
    vector<string> path;

    void dfs(int cur) {
        if (cur == n_) {
            ans.emplace_back(path);
            return ;
        }

        for (int i = 0; i < n_; i++) {
            if (!col[i] && !dg[n_ + cur - i] && !udg[i + cur]) {
                col[i] = dg[n_ + cur - i] = udg[i + cur] = true; 
                path[cur][i] = 'Q';
                dfs(cur + 1);
                path[cur][i] = '.';
                col[i] = dg[n_ + cur - i] = udg[i + cur] = false; 
            }
        }
    }

public:
    vector<vector<string>> solveNQueens(int n) {
        n_ = n;
        col = vector<bool>(n);
        dg = udg = vector<bool>(2 * n);
        path = vector<string>(n, string(n, '.'));
        dfs(0);
        return ans;
    }
};

93. 复原 IP 地址

输入:s = "25525511135" 输出:["255.255.11.135","255.255.111.35"]

class Solution {
    vector<string> ans;

    void dfs(const string &s, int cur, int cnt, string path) {
        if (cur == s.size() && cnt == 4) {
            ans.emplace_back(path.substr(0, path.size() - 1));
            return ;
        }

        if (cnt > 4) {
            return ;
        }

        for (int len = 1; len <= 3 && cur + len <= s.size(); len++) {
            string num = s.substr(cur, len);
            if (stoi(num) > 255 || (num[0] == '0' && num.size() > 1)) {
                continue;
            }
            dfs(s, cur + len, cnt + 1, path + num + ".");
        }
    }

public:
    vector<string> restoreIpAddresses(string s) {
        dfs(s, 0, 0, "");
        return ans;
    }
};

二分查找

35. 搜索插入位置

输入: nums = [1,3,5,6], target = 2
输出: 1
输入: nums = [1,3,5,6], target = 7
输出: 4

class Solution {
public:
    int searchInsert(vector<int>& nums, int target) {
        int left = 0, right = nums.size() - 1;
        while (left < right) {
            int mid = left + right >> 1;
            if (nums[mid] >= target) {
                right = mid;
            } else {
                left = mid + 1;
            }
        }
        return target > nums[left] ? left + 1 : left;
    }
};

69. x 的平方根

class Solution {
public:
    // 二分查找
    int mySqrt(int x) {
        if (x == 0 || x == 1) {
            return x;
        }

        int left = 0, right = x;
        while (left < right) {
            int mid = (left + right) >> 1;
            if (mid > x / mid) {
                right = mid;   
            } else {
                left = mid + 1;
            }
        }
        return left - 1;
    }
    // 牛顿迭代法
    int mySqrt(int x) {
        if (x == 0) {
            return 0;
        }

        double C = x, x0 = x;
        while (true) {
            double xi = 0.5 * (x0 + C / x0);
            if (fabs(x0 - xi) < 1e-7) {
                break;
            }
            x0 = xi;
        }
        return int(x0);
    }
};

74. 搜索二维矩阵

给你一个满足下述两条属性的 m x n 整数矩阵:

  • 每行中的整数从左到右按非严格递增顺序排列。
  • 每行的第一个整数大于前一行的最后一个整数。

给你一个整数 target ,如果 target 在矩阵中,返回 true ;否则,返回 false 。

class Solution {
public:
    bool searchMatrix(vector<vector<int>>& matrix, int target) {
        int m = matrix.size(), n = matrix[0].size();
        int left = 0, right = m * n - 1;
        while (left < right) {
            int mid = left + right >> 1;
            if (matrix[mid / n][mid % n] >= target) {
                right = mid;
            } else {
                left = mid + 1;
            }
        }
        return matrix[left / n][left % n] == target;
    }
};

240. 搜索二维矩阵 II

编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target 。该矩阵具有以下特性:

  • 每行的元素从左到右升序排列。
  • 每列的元素从上到下升序排列。
class Solution {
public:
    bool searchMatrix(vector<vector<int>>& matrix, int target) {
        int m = matrix.size(), n = matrix[0].size();
        int i = 0, j = n - 1;
        while (i < m && j >= 0) {
            if (matrix[i][j] == target) {
                return true;
            } else if (matrix[i][j] < target) {
                i++;
            } else {
                j--;
            }
        }
        return false;
    }
};

34. 在排序数组中查找元素的第一个和最后一个位置

class Solution {
public:
    vector<int> searchRange(vector<int>& nums, int target) {
        if (nums.empty()) {
            return {-1, -1};
        }

        vector<int> ans;
        int left = 0, right = nums.size() - 1;
        while (left < right) {
            int mid = left + right >> 1;
            if (nums[mid] >= target) {
                right = mid;
            } else {
                left = mid + 1;
            }
        }
        if (nums[left] != target) {
            return {-1, -1};
        }
        ans.emplace_back(left);

        left = 0, right = nums.size() - 1;
        while (left < right) {
            int mid = left + right + 1 >> 1;
            if (nums[mid] <= target) {
                left = mid;
            } else {
                right = mid - 1;
            }
        }
        ans.emplace_back(left);
        return ans;
    }
};

33. 搜索旋转排序数组

输入:nums = [4,5,6,7,0,1,2], target = 0
输出:4

class Solution {
public:
    int search(vector<int>& nums, int target) {
        int left = 0, right = nums.size() - 1;
        while (left < right) {
            int mid = left + right + 1 >> 1;
            if (nums[mid] > nums[0]) {
                left = mid;
            } else {
                right = mid - 1;
            }
        }
        if (target >= nums[0]) {
            left = 0;
        } else {
            left = left + 1;
            right = nums.size() - 1;
        }
        if (left > right) {
            return - 1;
        }
        while (left < right) {
            int mid = left + right + 1 >> 1;
            if (nums[mid] <= target) {
                left = mid;
            } else {
                right = mid - 1;
            }
        }
        return nums[left] == target ? left : -1;
    }
};

153. 寻找旋转排序数组中的最小值

class Solution {
public:
    int findMin(vector<int>& nums) {
        int left = 0, right = nums.size() - 1;
        while (left < right) {
            int mid = left + right >> 1;
            if (nums[mid] < nums[0]) {
                right = mid;
            } else {
                left = mid + 1;
            }
        }
        return min(nums[left], nums[0]);
    }
};

4. 寻找两个正序数组的中位数

给定两个大小分别为 m 和 n 的正序(从小到大)数组 nums1 和 nums2。请你找出并返回这两个正序数组的 中位数 。算法的时间复杂度应该为 O(log (m+n)) 。

class Solution {
    int getKthElement(vector<int> &nums1, vector<int> &nums2, int k) {
        int m = nums1.size();
        int n = nums2.size();
        int index1 = 0, index2 = 0;

        while (true) {
            if (index1 == m) {
                return nums2[index2 + k];
            }
            if (index2 == n) {
                return nums1[index1 + k];
            }
            if (k == 0) {
                return min(nums1[index1], nums2[index2]);
            }

            int newIndex1 = min(index1 + (k + 1) / 2 - 1, m - 1);
            int newIndex2 = min(index2 + (k + 1) / 2 - 1, n - 1);
            if (nums1[newIndex1] <= nums2[newIndex2]) {
                k -= newIndex1 - index1 + 1;
                index1 = newIndex1 + 1;
            } else {
                k -= newIndex2 - index2 + 1;
                index2 = newIndex2 + 1;
            }
        }
    }
    
public:
    double findMedianSortedArrays(vector<int>& nums1, vector<int>& nums2) {
        int totalLength = nums1.size() + nums2.size();
        if (totalLength % 2) {
            return getKthElement(nums1, nums2, totalLength / 2);
        } else {
            int left = getKthElement(nums1, nums2, totalLength / 2 - 1);
            int right = getKthElement(nums1, nums2, totalLength / 2);
            return (left + right) / 2.0;
        }
    }
};

20. 有效的括号

输入:s = "()[]{}"
输出:true

class Solution {
public:
    bool isValid(string s) {
        stack<char> stk;

        unordered_map<char, char> table = {
            {'(', ')'},
            {'{', '}'},
            {'[', ']'}
        };

        for (const auto &ch : s) {
            if (table.find(ch) != table.end()) {
                stk.push(ch);
            } else {
                if (!stk.empty() && table[stk.top()] == ch) {
                    stk.pop();
                } else {
                    return false;
                }
            }
        }
        return stk.empty();
    }
};

155. 最小栈

// MinStack minStack = new MinStack();
// minStack.push(-2);
// minStack.push(0);
// minStack.push(-3);
// minStack.getMin();   --> 返回 -3.
// minStack.pop();
// minStack.top();      --> 返回 0.
// minStack.getMin();   --> 返回 -2.
class MinStack {
    stack<int> stk, f;
public:
    MinStack() {

    }
    
    void push(int val) {
        stk.push(val);
        if (f.empty() || val <= f.top()) {
            f.push(val);
        }
    }
    
    void pop() {
        if (stk.top() == f.top()) {
            f.pop();
        }
        stk.pop();
    }
    
    int top() {
        return stk.top();
    }
    
    int getMin() {
        return f.top();
    }
};

162. 寻找峰值

给你一个整数数组 nums,找到峰值元素并返回其索引。数组可能包含多个峰值,在这种情况下,返回 任何一个峰值 所在位置即可。 你可以假设 nums[-1] = nums[n] = -∞ 。

class Solution {
public:
    int findPeakElement(vector<int>& nums) {
        int left = 0, right = nums.size() - 1;
        while (left < right) {
            int mid = (left + right + 1) >> 1;
            if (nums[mid] > nums[mid - 1]) {
                left = mid;
            } else {
                right = mid - 1;
            }
        }
        return left;
    }
};

232. 用栈实现队列

// ["MyQueue", "push", "push", "peek", "pop", "empty"]
// [[], [1], [2], [], [], []]
// 输出:
// [null, null, null, 1, 1, false]

// 解释:
// MyQueue myQueue = new MyQueue();
// myQueue.push(1); // queue is: [1]
// myQueue.push(2); // queue is: [1, 2] (leftmost is front of the queue)
// myQueue.peek(); // return 1
// myQueue.pop(); // return 1, queue is [2]
// myQueue.empty(); // return false
class MyQueue {
    stack<int> in, out;

public:
    MyQueue() {

    }

    void in2out() {
        while (in.size()) {
            out.push(in.top());
            in.pop();
        }
    }
    
    void push(int x) {
        in.push(x);
    }
    
    int pop() {
        if (out.empty()) {
            in2out();
        }
        int front = out.top();
        out.pop();
        return front;
    }
    
    int peek() {
        if (out.empty()) {
            in2out();
        }
        return out.top();
    }
    
    bool empty() {
        return in.empty() && out.empty();
    }
};

394. 字符串解码

输入:s = "3[a]2[bc]"
输出:"aaabcbc"

class Solution {
    string dfs(const string &s, int &cur) {
        string ans;
        while (cur < s.size() && s[cur] != ']') {
            if (s[cur] >= 'a' && s[cur] <= 'z') {
                ans += s[cur++];
            } else {
                int k = cur;
                while (s[k] >= '0' && s[k] <= '9') k++;
                int num = stoi(s.substr(cur, k - cur));
                cur = k + 1;
                string sub = dfs(s, cur);
                cur++;
                while (num--) {
                    ans += sub;
                }
            }
        }
        return ans;
    }

public:
    string decodeString(string s) {
        int cur = 0;
        return dfs(s, cur);
    }
};

739. 每日温度

给定一个整数数组 temperatures ,表示每天的温度,返回一个数组 answer ,其中 answer[i] 是指对于第 i 天,下一个更高温度出现在几天后。如果气温在这之后都不会升高,请在该位置用 0 来代替。
输入: temperatures = [73,74,75,71,69,72,76,73]
输出: [1,1,4,2,1,1,0,0]

class Solution {
public:
    vector<int> dailyTemperatures(vector<int>& temperatures) {
        stack<int> stk;
        vector<int> ans(temperatures.size());
        for (int i = temperatures.size() - 1; i >= 0; i--) {
            while (!stk.empty() && temperatures[stk.top()] <= temperatures[i]) {
                stk.pop();
            }
            if (stk.empty()) {
                ans[i] = 0;
            } else {
                ans[i] = stk.top() - i;
            }
            stk.push(i);
        }
        return ans;
    }
};

84. 柱状图中最大的矩形

给定 n 个非负整数,用来表示柱状图中各个柱子的高度。每个柱子彼此相邻,且宽度为 1 。
求在该柱状图中,能够勾勒出来的矩形的最大面积。

class Solution {
public:
    int largestRectangleArea(vector<int>& heights) {
        int n = heights.size();
        vector<int> left(n), right(n);
        stack<int> stk;
        for (int i = 0; i < heights.size(); i++) {
            while (!stk.empty() && heights[stk.top()] >= heights[i]) {
                stk.pop();
            }
            if (stk.empty()) {
                left[i] = -1;
            } else {
                left[i] = stk.top();
            }
            stk.push(i);
        }

        stk = stack<int>();
        for (int i = heights.size() - 1; i >= 0; i--) {
            while (!stk.empty() && heights[stk.top()] >= heights[i]) {
                stk.pop();
            }
            if (stk.empty()) {
                right[i] = n;
            } else {
                right[i] = stk.top();
            }
            stk.push(i);
        }

        int ans = 0;
        for (int i = 0; i < heights.size(); i++) {
            ans = max(ans, (right[i] - left[i] - 1) * heights[i]);
        }
        return ans;
    }
};

85. 最大矩形

给定一个仅包含 0 和 1 、大小为 rows x cols 的二维二进制矩阵,找出只包含 1 的最大矩形,并返回其面积。

class Solution {
public:
    int maximalRectangle(vector<vector<char>>& matrix) {
        int m = matrix.size();
        int n = matrix[0].size();

        vector<vector<int>> h(m, vector<int>(n));
        for (int i = 0; i < m; i++) {
            for (int j = 0; j < n; j++) {
                if (matrix[i][j] == '1') {
                    if (i == 0) {
                        h[i][j] = 1;
                    } else {
                        h[i][j] = h[i - 1][j] + 1;
                    }
                }
            }
        }

        int ans = 0;
        for (int i = 0; i < m; i++) {
            ans = max(ans, largestRectangleArea(h[i]));
        }
        return ans;
    }
};

215. 数组中的第K个最大元素

输入: [3,2,1,5,6,4], k = 2
输出: 5

class Solution {
    int quick_sort(vector<int> &nums, int left, int right, int k) {
        if (left >= right) {
            return nums[k];
        }

        int i = left - 1, j = right + 1, x = nums[left + right >> 1];
        while (i < j) {
            do i++; while (nums[i] < x);
            do j--; while (nums[j] > x);
            if (i < j) swap(nums[i], nums[j]);
        }
        if (k <= j) return quick_sort(nums, left, j, k);
        else return quick_sort(nums, j + 1, right, k);
    }

public:
    int findKthLargest(vector<int>& nums, int k) {
        return quick_sort(nums, 0, nums.size() - 1, nums.size() - k);
    }
};

347. 前 K 个高频元素

输入: nums = [1,1,1,2,2,3], k = 2
输出: [1,2]

class Solution {
public:
    vector<int> topKFrequent(vector<int>& nums, int k) {
        unordered_map<int, int> cnt;
        for (const auto &num : nums) {
            cnt[num]++;
        }

        vector<int> s(nums.size() + 1);
        for (const auto &item : cnt) {
            s[item.second]++;
        }

        int i = nums.size(), sum = 0;
        for (; i >= 0; i--) {
            sum += s[i];
            if (sum > k) {
                break;
            }
        }

        vector<int> ans;
        for (const auto &item : cnt) {
            if (item.second > i) {
                ans.emplace_back(item.first);
            }
        }
        return ans;
    }
};

295. 数据流的中位数

class MedianFinder {
    priority_queue<int, vector<int>, greater<int>> up;
    priority_queue<int> down;

public:
    MedianFinder() {}
    
    void addNum(int num) {
        if (down.empty() || num <= down.top()) {
            down.push(num);
            if (down.size() > up.size() + 1) {
                up.push(down.top());
                down.pop();
            }
        } else {
            up.push(num);
            if (up.size() > down.size()) {
                down.push(up.top());
                up.pop();
            }
        }
    }
    
    double findMedian() {
        if ((up.size() + down.size()) % 2) {
            return down.top();
        } else {
            return (up.top() + down.top()) / 2.0;
        }
    }
};

贪心算法

121. 买卖股票的最佳时机

你只能选择 某一天 买入这只股票,并选择在 未来的某一个不同的日子 卖出该股票。设计一个算法来计算你所能获取的最大利润。

class Solution {
public:
    int maxProfit(vector<int>& prices) {
        int ans = 0;
        for (int i = 0, minPrice = INT_MAX; i < prices.size(); i++) {
            ans = max(ans, prices[i] - minPrice);
            minPrice = min(minPrice, prices[i]);
        }
        return ans;
    }
};

122. 买卖股票的最佳时机 II

在每一天,你可以决定是否购买和/或出售股票。你在任何时候 最多 只能持有 一股 股票。你也可以先购买,然后在 同一天 出售。

class Solution {
public:
    int maxProfit(vector<int>& prices) {
        int f = 0; // 手里没股票
        int g = -prices[0]; // 手里有股票
        for (int i = 1; i < prices.size(); i++) {
            int newf = max(f, g + prices[i]);
            int newg = max(f - prices[i], g);
            f = newf;
            g = newg;
        }
        return f;
    }
};

309. 最佳买卖股票时机含冷冻期

卖出股票后,你无法在第二天买入股票 (即冷冻期为 1 天)。

class Solution {
public:
    int maxProfit(vector<int>& prices) {
        // 冷冻期 0
        // 已买入 1
        // 卖出   2
        int n = prices.size();
        vector<vector<int>> f(n, vector<int>(3, -INT_MAX));
        f[0][0] = 0;
        f[0][1] = -prices[0];
        for (int i = 1; i < n; i++) {
            f[i][0] = max(f[i - 1][0], f[i - 1][2]);
            f[i][1] = max(f[i - 1][0] - prices[i], f[i - 1][1]);
            f[i][2] = f[i - 1][1] + prices[i];
        }
        return max(f[n - 1][0], max(f[n - 1][1], f[n - 1][2]));
    }
};

55. 跳跃游戏

输入:nums = [2,3,1,1,4]
输出:true
解释:可以先跳 1 步,从下标 0 到达下标 1, 然后再从下标 1 跳 3 步到达最后一个下标。

class Solution {
public:
    bool canJump(vector<int>& nums) {
        for (int i = 0, j = 0; i < nums.size(); i++) {
            if (i > j) {
                return false;
            }
            j = max(j, nums[i] + i);
        }
        return true;
    }
};

45. 跳跃游戏 II

输入: nums = [2,3,1,1,4]
输出: 2
解释: 跳到最后一个位置的最小跳跃数是 2。从下标为 0 跳到下标为 1 的位置,跳 1 步,然后跳 3 步到达数组的最后一个位置。

class Solution {
public:
    int jump(vector<int>& nums) {
        vector<int> f(nums.size());
        for (int i = 1, j = 0; i < nums.size(); i++) {
            while (j + nums[j] < i) {
                j++;
            }
            f[i] = f[j] + 1;
        }
        return f[nums.size() - 1];
    }
};

763. 划分字母区间

给你一个字符串 s 。我们要把这个字符串划分为尽可能多的片段,同一字母最多出现在一个片段中。
注意,划分结果需要满足:将所有划分结果按顺序连接,得到的字符串仍然是 s 。
返回一个表示每个字符串片段的长度的列表。 输入:s = "ababcbacadefegdehijhklij"
输出:[9,7,8]
解释:划分结果为 "ababcbaca"、"defegde"、"hijhklij" 。每个字母最多出现在一个片段中。像 "ababcbacadefegde", "hijhklij" 这样的划分是错误的,因为划分的片段数较少。

class Solution {
public:
    vector<int> partitionLabels(string s) {
        unordered_map<char, int> last;
        for (int i = 0; i < s.size(); i++) {
            last[s[i]] = i;
        }
        vector<int> ans;
        int start = 0, end = 0;
        for (int i = 0; i < s.size(); i++) {
            end = max(end, last[s[i]]);
            if (i == end) {
                ans.emplace_back(end - start + 1);
                end = start = i + 1;
            }
        }
        return ans;
    }
};

动态规划

70. 爬楼梯

class Solution {
    using LL = long long;

public:
    int climbStairs(int n) {
        LL a = 1, b = 1;
        while (n--) {
            int c = a + b;
            a = b;
            b = c;
        }
        return a;
    }
};

96. 不同的二叉搜索树

给你一个整数 n ,求恰由 n 个节点组成且节点值从 1 到 n 互不相同的 二叉搜索树 有多少种?返回满足题意的二叉搜索树的种数。

class Solution {
public:
    int numTrees(int n) {
        vector<int> f(n + 1, 0);
        f[0] = 1;
        for (int i = 1; i <= n; i++) {
            for (int left = 0; left <= i - 1; left++) {
                int right = i - 1 - left;
                f[i] += f[left] * f[right];
            }
        }
        return f[n];
    }
};

118. 杨辉三角

class Solution {
public:
    vector<vector<int>> generate(int numRows) {
        vector<vector<int>> ans;
        for (int i = 0; i < numRows; i++) {
            vector<int> line(i + 1);
            line[0] = line[i] = 1;
            for (int j = 1; j < i; j++) {
                line[j] = ans[i - 1][j - 1] + ans[i - 1][j];
            }
            ans.emplace_back(line);
        }
        return ans;
    }
};

198. 打家劫舍

输入:[1,2,3,1] 输出:4 解释:偷窃 1 号房屋 (金额 = 1) ,然后偷窃 3 号房屋 (金额 = 3)。偷窃到的最高金额 = 1 + 3 = 4 。

class Solution {
public:
    int rob(vector<int>& nums) {
        if (nums.size() == 1) {
            return nums[0];
        }
        
        vector<int> f(nums.size());
        f[0] = nums[0];
        f[1] = max(nums[0], nums[1]);
        for (int i = 2; i < nums.size(); i++) {
            f[i] = max(f[i - 2] + nums[i], f[i - 1]);
        }
        return f[nums.size() - 1];
    }
};

213. 打家劫舍 II

所有的房屋都 围成一圈

class Solution {
    int robRange(vector<int> &nums, int start, int end) {
        vector<int> f(nums.size());
        f[start] = nums[start];
        f[start + 1] = max(nums[start], nums[start + 1]);
        for (int i = start + 2; i <= end; i++) {
            f[i] = max(f[i - 2] + nums[i], f[i - 1]);
        }
        return max(f[end - 1], f[end]);
    }

public:
    int rob(vector<int>& nums) {
        if (nums.size() == 1) {
            return nums[0];
        }
        if (nums.size() == 2) {
            return max(nums[0], nums[1]);
        }
        return max(robRange(nums, 0, nums.size() - 2), robRange(nums, 1, nums.size() - 1));
    }
};

337. 打家劫舍 III

class Solution {
    vector<int> dfs(TreeNode *node) {
        if (node == nullptr) {
            return {0, 0};
        }
        auto left = dfs(node->left);
        auto right = dfs(node->right);
        int select = node->val + left[0] + right[0];
        int noSelect = max(left[0], left[1]) + max(right[0], right[1]);
        return {noSelect, select};
    }

public:
    int rob(TreeNode* root) {
        auto ans = dfs(root);
        return max(ans[0], ans[1]);
    }
};

279. 完全平方数

给你一个整数 n ,返回 和为 n 的完全平方数的最少数量 。
完全平方数 是一个整数,其值等于另一个整数的平方;换句话说,其值等于一个整数自乘的积。例如,1、4、9 和 16 都是完全平方数,而 3 和 11 不是。
输入:n = 13
输出:2
解释:13 = 4 + 9

class Solution {
public:
    int numSquares(int n) {
        vector<int> f(n + 1, n);
        for (int i = 1; i <= sqrt(n); i++) {
            f[i * i] = 1;
        }
        for (int i = 2; i <= n; i++) {
            for (int j = 1; j <= sqrt(i); j++) {
                f[i] = min(f[i], f[i - j * j] + 1);
            }
        }
        return f[n];
    }
};

322. 零钱兑换

输入:coins = [1, 2, 5], amount = 11
输出:3
解释:11 = 5 + 5 + 1

class Solution {
public:
    int coinChange(vector<int>& coins, int amount) {
        vector<int> f(amount + 1, INT_MAX - 1);
        f[0] = 0;
        for (const auto &v : coins) {
            for (int j = v; j <= amount; j++) {
                f[j] = min(f[j], f[j - v] + 1);
            }
        }
        return f[amount] == INT_MAX - 1 ? -1 : f[amount];
    }
};

139. 单词拆分

给你一个字符串 s 和一个字符串列表 wordDict 作为字典。如果可以利用字典中出现的一个或多个单词拼接出 s 则返回 true。
注意:不要求字典中出现的单词全部都使用,并且字典中的单词可以重复使用。

class Solution {
public:
    bool wordBreak(string s, vector<string>& wordDict) {
        unordered_set<string> wordSet(wordDict.begin(), wordDict.end());
        int n = s.size();
        vector<int> f(n + 1);
        f[0] = true;
        s = " " + s;
        for (int i = 1; i <= n; i++) {
            for (int j = 1; j <= i; j++) {
                if (f[j - 1] && wordSet.find(s.substr(j, i - j + 1)) != wordSet.end()) {
                    f[i] = true;
                }
            }
        }
        return f[n];
    }
};

300. 最长递增子序列

输入:nums = [10,9,2,5,3,7,101,18]
输出:4
解释:最长递增子序列是 [2,3,7,101],因此长度为 4 。

class Solution {
public:
    int lengthOfLIS(vector<int>& nums) {
        vector<int> q;
        for (const auto &num : nums) {
            if (q.empty() || q.back() < num) {
                q.emplace_back(num);
            } else {
                if (num <= q[0]) {
                    q[0] = num;
                } else {
                    int left = 0, right = q.size() - 1;
                    while (left < right) {
                        int mid = (left + right + 1) >> 1;
                        if (q[mid] < num) {
                            left = mid;
                        } else {
                            right = mid - 1;
                        }
                    }
                    q[left + 1] = num;
                }
            }
        }
        return q.size();
    }
};

152. 乘积最大子数组

输入: nums = [2,3,-2,4]
输出: 6
解释: 子数组 [2,3] 有最大乘积 6。

class Solution {
public:
    int maxProduct(vector<int>& nums) {
        double ans = nums[0], f = nums[0], g = nums[0];
        for (int i = 1; i < nums.size(); i++) {
            double a = nums[i], fa = f * a, ga = g * a;
            f = max(a, max(fa, ga));
            g = min(a, min(fa, ga));
            ans = max(ans, f);
        }
        return ans;
    }
};

416. 分割等和子集

输入:nums = [1,5,11,5]
输出:true
解释:数组可以分割成 [1, 5, 5] 和 [11] 。

class Solution {
public:
    bool canPartition(vector<int>& nums) {
        if (nums.size() <= 1) {
            return false;
        }
        int sum = 0;
        for (const auto &num : nums) {
            sum += num;
        }
        if (sum % 2) {
            return false;
        }
        sum /= 2;

        vector<int> f(sum + 1);
        f[0] = true;
        for (const auto &v : nums) {
            for (int j = sum; j >= v; j--) {
                f[j] |= f[j - v];
            }
        }
        return f[sum];
    }
};

32. 最长有效括号

给你一个只包含 '(' 和 ')' 的字符串,找出最长有效(格式正确且连续)括号子串的长度。
输入:s = ")()())"
输出:4
解释:最长有效括号子串是 "()()"

class Solution {
public:
    int longestValidParentheses(string s) {
        int ans = 0;
        vector<int> f(s.size());
        for (int i = 1; i < s.size(); i++) {
            if (s[i] == ')') {
                if (s[i - 1] == '(') {
                    f[i] = (i - 2 >= 0 ? f[i - 2] : 0) + 2;
                } else if (i - f[i - 1] - 1 >= 0 && s[i - f[i - 1] - 1] == '(') {
                    f[i] = (i - f[i - 1] - 2 >= 0 ? f[i - f[i - 1] - 2] : 0) + f[i - 1] + 2;
                }
                ans = max(ans, f[i]);
            }
        }
        return ans;
    }
};

494. 目标和

给你一个非负整数数组 nums 和一个整数 target 。
向数组中的每个整数前添加 '+' 或 '-' ,然后串联起所有整数,可以构造一个 表达式
返回可以通过上述方法构造的、运算结果等于 target 的不同 表达式 的数目。

class Solution {
public:
    int findTargetSumWays(vector<int>& nums, int target) {
        int sum = 0;
        for (const auto &num : nums) {
            sum += num;
        }
        // sum - neg - neg = target
        // 2 * neg = sum - target;
        int neg = sum - target;
        if (neg < 0 || neg % 2) {
            return 0;
        }
        neg /= 2;
        vector<int> f(neg + 1);
        f[0] = 1;
        for (const auto &num : nums) {
            for (int j = neg; j >= num; j--) {
                f[j] += f[j - num];
            }
        }
        return f[neg];
    }
};

338. 比特位计数

class Solution {
public:
    vector<int> countBits(int n) {
        vector<int> f(n + 1);
        for (int i = 1; i <= n; i++) {
            f[i] = f[i >> 1] + (i & 1);
        }
        return f;
    }
};

多维动态规划

62. 不同路径

[0, 0] -> [m - 1, n - 1] 路径数量

class Solution {
public:
    int uniquePaths(int m, int n) {
        vector<vector<int>> f(m, vector<int>(n));
        for (int i = 0; i < m; i++) {
            for (int j = 0; j < n; j++) {
                if (!i && !j) f[i][j] = 1;
                else {
                    if (i) f[i][j] += f[i - 1][j];
                    if (j) f[i][j] += f[i][j - 1];
                }
            }
        }
        return f[m - 1][n - 1];
    }
};

64. 最小路径和

给定一个包含非负整数的 m x n 网格 grid ,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小。

class Solution {
public:
    int minPathSum(vector<vector<int>>& grid) {
        int m = grid.size(), n = grid[0].size();
        vector<vector<int>> f(m, vector<int>(n, INT_MAX));
        for (int i = 0; i < m; i++) {
            for (int j = 0; j < n; j++) {
                if (!i && !j) f[i][j] = grid[i][j];
                else {
                    if (i) f[i][j] = min(f[i][j], f[i - 1][j] + grid[i][j]);
                    if (j) f[i][j] = min(f[i][j], f[i][j - 1] + grid[i][j]);
                }
            }
        }
        return f[m - 1][n - 1];
    }
};

647. 回文子串

给你一个字符串 s ,请你统计并返回这个字符串中 回文子串 的数目。

class Solution {
public:
    int countSubstrings(string s) {
        vector<vector<int>> f(s.size(), vector<int>(s.size()));

        int cnt = 0;

        for (int j = 0; j < s.size(); j++) {
            for (int i = 0; i <= j; i++) {
                if (s[i] != s[j]) {
                    continue;
                }
                if (i == j || i +1 == j) {
                    f[i][j] = 1;
                } else {
                    f[i][j] = f[i + 1][j - 1];
                }
                if (f[i][j]) {
                    cnt++;
                }
            }
        }
        return cnt;
    }
};

5. 最长回文子串

class Solution {
};
class Solution {
public:
    // dp 方法,时间复杂度 O(n^2),空间复杂度 O(n^2)
    string longestPalindrome(string s) {
        int n = s.size();
        vector<vector<int>> f(n, vector<int>(n));
        string ans;
        for (int j = 0; j < n; j++) {
            for (int i = 0; i <= j; i++) {
                if (s[i] != s[j]) {
                    continue;
                }
                if (i == j || i + 1 == j) {
                    f[i][j] = true;
                } else {
                    f[i][j] = f[i + 1][j - 1];
                }

                if (f[i][j] && j - i + 1 > ans.size()) {
                    ans = s.substr(i, j - i + 1);
                }
            }
        }
        return ans;
    }
    // 中心扩展,时间复杂度 O(n^2),空间复杂度 O(1)
    string longestPalindrome(string s) {
        string ans;
        for (int i = 0; i < s.size(); i++) {
            int left = i - 1, right = i + 1;
            while (left >= 0 && right < s.size() && s[left] == s[right]) {
                left--;
                right++;
            }
            if (ans.size() < right - left - 1) {
                ans = s.substr(left + 1, right - left - 1);
            }

            left = i, right = i + 1;
            while (left >= 0 && right < s.size() && s[left] == s[right]) {
                left--;
                right++;
            }
            if (ans.size() < right - left - 1) {
                ans = s.substr(left + 1, right - left - 1);
            }
        }
        return ans;
    }
};

1143. 最长公共子序列

输入:text1 = "abcde", text2 = "ace"
输出:3
解释:最长公共子序列是 "ace" ,它的长度为 3 。

class Solution {
public:
    int longestCommonSubsequence(string text1, string text2) {
        int m = text1.size();
        int n = text2.size();
        vector<vector<int>> f(m + 1, vector<int>(n + 1));
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                f[i][j] = max(f[i - 1][j], f[i][j - 1]);
                int t = (text1[i - 1] == text2[j - 1]);
                f[i][j] = max(f[i][j], f[i - 1][j - 1] + t);
            }
        }
        return f[m][n];
    }
};

最长公共子串

class Solution {
public:
    string LCS(string str1, string str2) {
        int m = str1.size(), n = str2.size();
        vector<vector<int>> f(m + 1, vector<int>(n + 1));
        string ans;
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                if (str1[i - 1] != str2[j - 1]) {
                    f[i][j] = 0;
                } else {
                    f[i][j] = f[i - 1][j - 1] + 1;
                    if (f[i][j] > ans.size()) {
                        ans = str1.substr(i - f[i][j] + 1 - 1, f[i][j]);
                    }
                }
            }
        }
        return ans;
    }
};

718. 最长重复子数组

class Solution {
public:
    int findLength(vector<int>& nums1, vector<int>& nums2) {
        int m = nums1.size();
        int n = nums2.size();
        vector<vector<int>> f(m + 1, vector<int>(n + 1));
        int ans = 0;
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                if (nums1[i - 1] != nums2[j - 1]) {
                    f[i][j] = 0;
                } else {
                    f[i][j] = f[i - 1][j - 1] + 1;
                    ans = max(ans, f[i][j]);
                }
            }
        }
        return ans;
    }
};

72. 编辑距离

给你两个单词 word1 和 word2, 请返回将 word1 转换成 word2 所使用的最少操作数 你可以对一个单词进行如下三种操作:

  • 插入一个字符
  • 删除一个字符
  • 替换一个字符
class Solution {
public:
    int minDistance(string word1, string word2) {
        int m = word1.size(), n = word2.size();
        vector<vector<int>> f(m + 1, vector<int>(n + 1));
        for (int i = 1; i <= m; i++) {
            f[i][0] = i;
        }
        for (int j = 1; j <= n; j++) {
            f[0][j] = j;
        }
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                f[i][j] = min(f[i - 1][j], f[i][j - 1]) + 1;
                int t = (word1[i - 1] != word2[j - 1]);
                f[i][j] = min(f[i][j], f[i - 1][j - 1] + t);
            }
        }
        return f[m][n];
    }
};

221. 最大正方形

在一个由 '0' 和 '1' 组成的二维矩阵内,找到只包含 '1' 的最大正方形,并返回其面积。

class Solution {
public:
    int maximalSquare(vector<vector<char>>& matrix) {
        int m = matrix.size();
        int n = matrix[0].size();
        vector<vector<int>> f(m + 1, vector<int>(n + 1));
        int ans = 0;
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                if (matrix[i - 1][j - 1] == '0') {
                    continue;
                }
                f[i][j] = min(f[i - 1][j], min(f[i][j - 1], f[i - 1][j - 1])) + 1;
                ans = max(ans, f[i][j]);
            }
        }
        return ans * ans;
    }
};

312. 戳气球

// 输入:nums = [3,1,5,8]
// 输出:167
// 解释:
// nums = [3,1,5,8] --> [3,5,8] --> [3,8] --> [8] --> []
// coins =  3*1*5    +   3*5*8   +  1*3*8  + 1*8*1 = 167
class Solution {
public:
    int maxCoins(vector<int>& nums) {
        int n = nums.size();
        vector<int> a(n + 2, 1);
        for (int i = 1; i <= n; i++) {
            a[i] = nums[i - 1];
        }

        vector<vector<int>> f(n + 2, vector<int>(n + 2));
        for (int len = 3; len <= n + 2; len++) {
            for (int i = 0; i + len - 1 < n + 2; i++) {
                int j = i + len - 1;
                for (int k = i + 1; k < j; k++) {
                    f[i][j] = max(f[i][j], f[i][k] + f[k][j] + a[i] * a[k] * a[j]);
                }
            }
        }
        return f[0][n + 1];
    }
};

10. 正则表达式匹配

class Solution {
public:
    bool isMatch(string s, string p) {
        int m = s.size();
        int n = p.size();
        s = ' ' + s;
        p = ' ' + p;
        vector<vector<bool>> f(m + 1, vector<bool>(n + 1));
        f[0][0] = true;
        for (int j = 2; j <= n; j += 2) {
            if (p[j] == '*') {
                f[0][j] = f[0][j - 2];
            }
        }
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                if (j + 1 <= n && p[j + 1] == '*') {
                    continue;
                }
                if (s[i] == p[j] || p[j] == '.') {
                    f[i][j] = f[i - 1][j - 1];
                } else if (p[j] == '*') {
                    if (s[i] == p[j - 1] || p[j - 1] == '.') {
                        f[i][j] = f[i][j - 2] || f[i - 1][j];
                    } else {
                        f[i][j] = f[i][j - 2];
                    }
                }
            }
        }
        return f[m][n];
    }
};

技巧

461. 汉明距离

两个整数之间的 汉明距离 指的是这两个数字对应二进制位不同的位置的数目。
给你两个整数 x 和 y,计算并返回它们之间的汉明距离。

class Solution {
public:
    int hammingDistance(int x, int y) {
        int num = x ^ y;
        int ans = 0;
        while (num) {
            ans += num & 1;
            num >>= 1;
        }
        return ans;
    }
};

136. 只出现一次的数字

给你一个 非空 整数数组 nums ,除了某个元素只出现一次以外,其余每个元素均出现两次。找出那个只出现了一次的元素。
你必须设计并实现线性时间复杂度的算法来解决此问题,且该算法只使用常量额外空间。

class Solution {
public:
    int singleNumber(vector<int>& nums) {
        int ans = 0;
        for (const auto &num : nums) {
            ans ^= num;
        }
        return ans;
    }
};

169. 多数元素

给定一个大小为 n 的数组 nums ,返回其中的多数元素。多数元素是指在数组中出现次数 大于 ⌊ n/2 ⌋ 的元素。
你可以假设数组是非空的,并且给定的数组总是存在多数元素。

class Solution {
public:
    int majorityElement(vector<int>& nums) {
        int candidate = 0, cnt = 0;
        for (const auto &num : nums) {
            if (cnt == 0) {
                candidate = num;
                cnt = 1;
            } else if (candidate != num) {
                cnt--;
            } else {
                cnt++;
            }
        }
        return candidate;
    }
};

75. 颜色分类

给定一个包含红色、白色和蓝色、共 n 个元素的数组 nums ,原地对它们进行排序,使得相同颜色的元素相邻,并按照红色、白色、蓝色顺序排列。
我们使用整数 0、 1 和 2 分别表示红色、白色和蓝色。

class Solution {
public:
    void sortColors(vector<int>& nums) {
        for (int left = 0, mid = 0, right = nums.size() - 1; mid <= right; ) {
            if (nums[mid] == 0) {
                swap(nums[left++], nums[mid++]);
            } else if (nums[mid] == 1) {
                mid++;
            } else {
                swap(nums[mid], nums[right--]);
            }
        }
    }
};

31. 下一个排列

必须 原地 修改,只允许使用额外常数空间。

class Solution {
public:
    void nextPermutation(vector<int>& nums) {
        int k = nums.size() - 1;
        while (k - 1 >= 0 && nums[k - 1] >= nums[k]) {
            k--;
        }
        if (k == 0) {
            reverse(nums.begin() + k, nums.end());
            return ;
        }

        int j = nums.size() - 1;
        while (j >= k && nums[k - 1] >= nums[j]) {
            j--;
        }
        swap(nums[k - 1], nums[j]);
        reverse(nums.begin() + k, nums.end());
    }
};

287. 寻找重复数

给定一个包含 n + 1 个整数的数组 nums ,其数字都在 [1, n] 范围内(包括 1 和 n),可知至少存在一个重复的整数。

class Solution {
public:
    int findDuplicate(vector<int>& nums) {
        int slow = 0, fast = 0;
        while (true) {
            slow = nums[slow];
            fast = nums[nums[fast]];
            if (slow == fast) {
                break;
            }
        }
        slow = 0;
        while (slow != fast) {
            slow = nums[slow];
            fast = nums[fast];
        }
        return slow;
    }
};

470. 用 Rand7() 实现 Rand10()

class Solution {
public:
    int rand10() {
        int t = (rand7() - 1) * 7 + rand7();
        if (t > 40) {
            return rand10();
        }
        return t % 10 + 1;
    }
};

621. 任务调度器

// 输入:tasks = ["A","A","A","B","B","B"], n = 2
// 输出:8
// 解释:A -> B -> (待命) -> A -> B -> (待命) -> A -> B
//      在本示例中,两个相同类型任务之间必须间隔长度为 n = 2 的冷却时间,而执行一个任务只需要一个单位时间,所以中间出现了(待命)状态。 
class Solution {
public:
    int leastInterval(vector<char>& tasks, int n) {
        unordered_map<char, int> table;
        for (const auto &ch : tasks) {
            table[ch]++;
        }

        int maxExec = 0;
        for (const auto &item : table) {
            maxExec = max(maxExec, item.second);
        }
        int maxCount = 0;
        for (const auto &item : table) {
            if (item.second == maxExec) {
                maxCount++;
            }
        }
        return max((maxExec - 1) * (n + 1) + maxCount, (int)tasks.size());
    }
};

找出系统中内存占用最大的时刻,此时的内存占用是多少,此时系统中都有哪些内存的 id

#include <iostream>
#include <vector>
#include <algorithm>
#include <set>

using namespace std;

struct memRequest {
    int id;    // 请求 ID
    int mem;   // 请求的内存大小
    int start; // 开始时间
    int end;   // 结束时间
};

struct Event {
    int time;
    int memChange; // 内存变化量
    int id;

    // 自定义排序规则:先按时间排序;如果时间相同,start事件在前,end事件在后
    bool operator<(const Event &other) const {
        return time < other.time || (time == other.time && memChange > other.memChange);
    }
};

int main() {
    vector<memRequest> requests = {
        {1, 10, 0, 5},
        {2, 15, 2, 7},
        {3, 20, 3, 8},
        {4, 25, 6, 10},
        {5, 30, 8, 12},
    };

    vector<Event> events;

    // 创建事件列表
    for (const auto &req : requests) {
        events.push_back({req.start, req.mem, req.id});
        events.push_back({req.end, -req.mem, req.id});
    }

    // 按时间排序事件
    sort(events.begin(), events.end());

    int maxMemory = 0;
    int currentMemory = 0;
    set<int> currentIDs;
    set<int> maxMemoryIDs;

    // 扫描线算法,遍历所有事件
    for (const auto &event : events) {
        currentMemory += event.memChange;

        if (event.memChange > 0) {
            currentIDs.insert(event.id);
        } else {
            currentIDs.erase(event.id);
        }

        // 更新最大内存使用量和对应的 ID 列表
        if (currentMemory > maxMemory) {
            maxMemory = currentMemory;
            maxMemoryIDs = currentIDs;
        }
    }

    // 输出结果
    std::cout << "Maximum Memory Usage: " << maxMemory << std::endl;
    std::cout << "Requests at that time: ";
    for (int id : maxMemoryIDs) {
        std::cout << id << " ";
    }
    std::cout << std::endl;

    return 0;
}