目录

LC_HOT100

目录

1.两数之和

题意

题解

哈希表

哈希表的key表示数值,value是个vector,表示下标

遍历每个数,分两种情况讨论。v + v == target和 v + a == target

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
class Solution {
public:
    vector<int> twoSum(vector<int>& nums, int target) {
        unordered_map<int, vector<int>> mp;
        for (int i = 0; i < (int) nums.size(); ++i) {
            mp[nums[i]].push_back(i);
        }
        for (int i = 0; i < (int) nums.size(); ++i) {
            if (nums[i] * 2 == target and mp[nums[i]].size() > 1) {
                return vector<int>{mp[nums[i]][0], mp[nums[i]][1]};
            } else if (nums[i] * 2 != target and mp.find(target - nums[i]) != mp.end()) {
                return vector<int>{mp[nums[i]][0], mp[target - nums[i]][0]};
            }
        }
        return vector<int>{};
    }
};

49.字母异位词分组

题意

题解

通过计数哈希,时间复杂度O(n(k + 26))

遍历每个字符串,用哈希表存每个字符出现的次数,遍历a-z,按顺序重新组装作为该字符串的hash值

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
class Solution {
public:
    vector<vector<string>> groupAnagrams(vector<string>& strs) {
        unordered_map<string, vector<int>> mp;
        for (int i = 0; i < (int) strs.size(); ++i) {
            string str = strs[i];
            unordered_map<char, int> mmp;
            for (char i : str) {
                mmp[i]++;
            }
            string tmp;
            for (char i = 'a'; i <= 'z'; ++i) {
                if (mmp.find(i) != mmp.end()) {
                    int tv = mmp[i];
                    while (tv--) {
                        tmp += i;
                    }
                }
            }
            mp[tmp].push_back(i);
        }
        vector<vector<string>> ans;
        for (auto& [x, y] : mp) {
            vector<string> now;
            for (int i : y) {
                now.push_back(strs[i]);
            }
            ans.push_back(now);
        }
        return ans;
    }
};

128.最长连续序列

题意

题解

哈希表,只从序列头开始计算,这样就不会重复计算

去重

遍历每个元素,假设值为v,如果找不到v-1,就说明他是序列头,那就一直找v v+1 v+2直到找不到

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
class Solution {
public:
    int longestConsecutive(vector<int>& nums) {
        unordered_set<int> us;
        for (int i : nums) {
            us.insert(i);
        }
        int ans = 0;
        for (int i : us) {
            if (us.find(i - 1) == us.end()) {   // 序列头
                int num = 0;
                for (int j = i; ; ++j) {
                    if (us.find(j) != us.end()) {
                        num++;
                    } else {
                        ans = max(ans, num);
                        break;
                    }
                }
            }
        }
        return ans;
    }
};

283.移动零

题意

题解

双指针,一个指针指向接下来的第一个0,一个指针指向接下来的第一个非0,然后交换

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
class Solution {
public:
    void moveZeroes(vector<int>& nums) {
        int n = (int) nums.size();
        if (n == 0) {
            return ;
        }
        int l = 0, r = 1;
        while (r < n) {
            if (nums[l] == 0 and nums[r] != 0) {
                swap(nums[l], nums[r]);
            }
            while (l < n) {
                if (nums[l] == 0) break;
                else l++;
            }
            r = l + 1;
            while (r < n) {
                if (nums[r] != 0) break;
                else r++;
            }
        }
    }
};

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

题意

题解

滑动窗口

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        int n = s.length();
        int l = 0, r = 1;
        if (n == 1) {
            return 1;
        }
        unordered_map<char, int> mp;
        mp[s[0]]++;
        int ans = 0;
        while (l < r and r <= n) {
            if (r - l > (int) mp.size()) {
                mp[s[l]]--;
                if (mp[s[l]] == 0) mp.erase(s[l]);
                l++;
            } else {
                ans = max(ans, r - l);
                if (r < n) {mp[s[r]]++; r++;}
                else break;
            }
        }
        return ans;
    }
};

560.和为K的子数组

题意

题解

前缀和

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
class Solution {
public:
    int subarraySum(vector<int>& nums, int k) {
        int n = (int) nums.size();
        int sum = 0;
        int ans = 0;
        unordered_map<int, int> mp;
        for (int i = 0; i < n; ++i) {
            sum += nums[i];
            int now = sum - k;
            ans += mp[now];
            mp[sum]++;
            if (sum == k) ans++;
        }
        return ans;
    }
};

53.最大子数组和

题意

题解

dp O(n)

dp[i]表示以i结尾的最大子数组和,dp[i] = max(dp[i -1] + a[i], a[i])

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
class Solution {
public:
    int maxSubArray(vector<int>& nums) {
        int dp = 0;
        int ans = -(1e4 + 3);
        for (int i : nums) {
            dp = max(dp + i, i);
            ans = max(ans, dp);
        }
        return ans;
    }
};

线段树

73.矩阵置零

题意

题解

要求空间复杂度为常数,就是把这一行要不要变成0的信息存储在ai0,把这一列要不要变成0的信息存储在a0i,然后用rowflag和colflag来判断最上面和最左边的行列是不是本来就有0

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
class Solution {
public:
    void setZeroes(vector<vector<int>>& matrix) {
        int n = (int) matrix.size();
        int m = (int) matrix[0].size();
        bool colflag = false;
        bool rowflag = false;
        for (int i = 0; i < n; ++i) {
            if (matrix[i][0] == 0) {
                colflag = true;
                break;
            }
        }
        for (int j = 0; j < m; ++j) {
            if (matrix[0][j] == 0) {
                rowflag = true;
                break;
            }
        }
        for (int i = 1; i < n; ++i) {
            for (int j = 1; j < m; ++j) {
                if (matrix[i][j] == 0) {
                    matrix[i][0] = matrix[0][j] = 0;
                }
            }
        }
        for (int i = 1; i < n; ++i) {
            if (matrix[i][0] == 0) {
                for (int j = 1; j < m; ++j) {
                    matrix[i][j] = 0;
                }
            }
        }
        for (int j = 1; j < m; ++j) {
            if (matrix[0][j] == 0) {
                for (int i = 1; i < n; ++i) {
                    matrix[i][j] = 0;
                }
            }
        }
        if (rowflag) {
            for (int j = 0; j < m; ++j) {
                matrix[0][j] = 0;
            }
        }
        if (colflag) {
            for (int i = 0; i < n; ++i) {
                matrix[i][0] = 0;
            }
        }
    }
};

160.相交链表

题意

题解

没什么好方法,干就完事了

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode(int x) : val(x), next(NULL) {}
 * };
 */
class Solution {
public:
    ListNode *getIntersectionNode(ListNode *headA, ListNode *headB) {
        int len1 = 0, len2 = 0;
        auto ptr1 = headA, ptr2 = headB;
        while (ptr1 != nullptr) {
            len1++;
            ptr1 = ptr1->next;
        }
        while (ptr2 != nullptr) {
            len2++;
            ptr2 = ptr2->next;
        }
        if (len1 > len2) {
            auto dif = len1 - len2;
            while (dif--) {
                headA = headA->next;
            }
        }
        if (len1 < len2) {
            auto dif = len2 - len1;
            while (dif--) {
                headB = headB->next;
            }
        }
        while (headA != nullptr) {
            if (headA == headB) {
                return headA;
            } else {
                headA = headA->next;
                headB = headB->next;
            }
        }
        return nullptr;
    }
};

94.二叉树的中序遍历

题意

题解

递归

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    vector<int> inorderTraversal(TreeNode* root) {
        vector<int> ans;
        function<void(TreeNode*)> dfs = [&](TreeNode* node) {
            if (node == nullptr) return;
            if (node->left != nullptr) dfs(node->left);
            ans.push_back(node->val);
            if (node->right != nullptr) dfs(node->right);
        };
        dfs(root);
        return ans;
    }
};

迭代

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    vector<int> inorderTraversal(TreeNode* root) {
        vector<int> ans;
        if (root == nullptr) return ans;
        stack<TreeNode*> st;
        auto cnt = root;
        while (cnt != nullptr || !st.empty()) { //!st.empty() 是关键
            while (cnt != nullptr) {	// 一直找左节点并入栈
                st.emplace(cnt);
                cnt = cnt->left;
            }
            cnt = st.top();
            st.pop();
            ans.push_back(cnt->val);	// 取出,打印节点
            cnt = cnt->right; //去右节点,然后再一直找左节点
        }
        return ans;
    }
};

200.岛屿数量

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
class Solution {
public:
    int numIslands(vector<vector<char>>& grid) {
        int n = (int) grid.size();
        int m = (int) grid[0].size();
        int ans = 0;
        int dir[][2] = {{0, 1}, {0, -1}, {1, 0}, {-1, 0}};
        function<void(int, int)> dfs = [&](int x, int y) {
            grid[x][y] = '2';
            for (int k = 0; k < 4; ++k) {
                int tx = x + dir[k][0], ty = y + dir[k][1];
                if (tx >= 0 and tx < n and ty >= 0 and ty < m and grid[tx][ty] == '1') dfs(tx, ty);
            }
        };
        for (int i = 0; i < n; ++i) {
            for (int j = 0; j < m; ++j) {
                if (grid[i][j] == '1') {
                    ans++;
                    dfs(i, j);
                }
            }
        }
        return ans;
    }
};

46.全排列

题意

题解

dfs,遍历到pos时,交换pos和之后的数,然后遍历下一个数,注意回溯

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
class Solution {
public:
    vector<vector<int>> permute(vector<int>& nums) {
        vector<vector<int>> ans;
        int n = (int) nums.size();
        function<void(int)> dfs = [&](int pos) {
            if (pos == n) {
                ans.emplace_back(nums);
                return ;
            }
            for (int i = pos; i < n; ++i) {	// 注意从pos开始遍历
                swap(nums[pos], nums[i]);
                dfs(pos + 1);	// 从下一个数开始
                swap(nums[pos], nums[i]);
            }
        };
        dfs(0);
        return ans;
    }
};

47.全排列II

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
class Solution {
public:
    vector<vector<int>> permuteUnique(vector<int>& nums) {
        vector<vector<int>> ans;
        int n = (int) nums.size();
        function<void(int)> dfs = [&](int pos) {
            if (pos == n) {
                ans.emplace_back(nums);
                return ;
            }
            unordered_set<int> s; //让它不出现5,6,6交换两次的情况,如果是5,5,5也不会重复交换
            for (int i = pos; i < n; ++i) {
                if (s.find(nums[i]) == s.end()) {
                    s.insert(nums[i]);
                    swap(nums[i], nums[pos]);
                    dfs(pos + 1);
                    swap(nums[i], nums[pos]);
                }
            }
        };
        dfs(0);
        return ans;
    }
};

LCR 016. 无重复字符的最长子串

题意

给定一个字符串 s ,请你找出其中不含有重复字符的 最长连续子字符串 的长度。

题解

尺取,注意边界

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        int n = s.length();
        int l = 0, r = 1;
        unordered_map<char, int> mp;
        mp[s[0]] = 1;
        int ans = 0;
        while (r <= n and l < r) {
            if (r - l != (int) mp.size()) {
                mp[s[l]]--;
                if (mp[s[l]] == 0) mp.erase(s[l]);
                l++;
            } else {
                ans = max(ans, r - l);
                //if (r < n) // infinite loop
                {mp[s[r]]++; r++;}
            }
        }
        return ans;
    }
};

146. LRU 缓存

题意

题解

使用双向链表,存一下sz和cap,判断是否需要逐出

用一个哈希表存key和对应节点的映射,可以快速找到节点

对于get操作,把节点移到头部 deleteNode addHead

对于put操作,如果存在,和get操作一样。如果不存在,生成一个节点,addHead,判断是否需要逐出,removeTail

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
class LRUCache {
private:
    struct DLinkedList {
        int key, value;
        DLinkedList* prev;
        DLinkedList* next;
        DLinkedList() : key(0), value(0), prev(nullptr), next(nullptr) {}
        DLinkedList(int k, int v) : key(k), value(v), prev(nullptr), next(nullptr) {}
    };
    unordered_map<int, DLinkedList*> mp;
    size_t sz;
    size_t cap;
    DLinkedList* head, *tail;
public:
    LRUCache(int capacity) : cap(capacity), sz(0) {
        head = new DLinkedList();
        tail = new DLinkedList();
        head->next = tail;
        tail->prev = head;
    }

    int get(int key) {
        if (mp.find(key) != mp.end()) {
            auto node = mp[key];
            deleteNode(node);
            addHead(node);
            return node->value;
        } else {
            return -1;
        }
    }

    void put(int key, int value) {
        if (mp.find(key) != mp.end()) {
            auto node = mp[key];
            node->value = value;
            deleteNode(node);
            addHead(node);
        } else {
            auto node = new DLinkedList(key, value);
            mp[key] = node;
            addHead(node);
            sz++;
            if (sz > cap) {
                removeTail();
                sz--;
            } 
        }
    }
    
    void removeTail() {
        auto node = tail->prev;
        node->prev->next = tail;
        tail->prev = node->prev;
        mp.erase(node->key);	// 别忘了这一步
        delete node;
    }
    
    void deleteNode(DLinkedList* node) {
        node->prev->next = node->next;
        node->next->prev = node->prev;
    }
    
    void addHead(DLinkedList* node) {
        node->next = head->next;
        node->prev = head;
        head->next->prev = node;
        head->next = node;
    }
};

/**
 * Your LRUCache object will be instantiated and called as such:
 * LRUCache* obj = new LRUCache(capacity);
 * int param_1 = obj->get(key);
 * obj->put(key,value);
 */

347. 前 K 个高频元素

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
class Solution {
public:
    vector<int> topKFrequent(vector<int>& nums, int k) {
        unordered_map<int, int> mp;
        for (int i : nums) {
            mp[i]++;
        }
        priority_queue<pair<int, int>, vector<pair<int, int>>, greater<>> pq;
        for (auto& [x, y] : mp) {
            if ((int) pq.size() < k) {
                pq.emplace(y, x);
            } else {
                if (y > pq.top().first) {
                    pq.pop();
                    pq.emplace(y, x);
                }
            }
        }
        vector<int> ans;
        while (!pq.empty()) {
            ans.emplace_back(pq.top().second);
            pq.pop();
        }
        return ans;
    }
};

692. 前K个高频单词

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
class Solution {
public:
    vector<string> topKFrequent(vector<string>& words, int k) {
        int n = (int) words.size();
        unordered_map<string, int> mp;
        for (const string& i : words) {
            mp[i]++;
        }
        struct cmp {
            bool operator() (const pair<string, int>& p1, const pair<string, int>& p2) const {
                if (p1.second == p2.second) return p1.first < p2.first;
                return p1.second > p2.second;
            }
        };
        priority_queue<pair<string, int>, vector<pair<string, int>>, cmp> pq;
        for (auto& [x, y] : mp) {
            pq.emplace(x, y);
            if ((int) pq.size() > k) {
                pq.pop();
            }
        }
        vector<string> ans;
        while (!pq.empty()) {
            auto& [x, _] = pq.top();
            ans.emplace_back(x);
            pq.pop();
        }
        reverse(ans.begin(), ans.end());
        return ans;
    }
};

25. K 个一组翻转链表

题意

题解

用四个指针,start指向将要翻转的这一组的头,end指向尾,pre指向start的前一个节点(需要在head前插一个虚拟的节点),nxt指向end的后一个节点

然后翻转这一组后就是各种指针移动(赋值)

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    void reverse(ListNode* head, ListNode* tail) {
        ListNode* l, *m, *r;
        l = nullptr;
        m = head;
        r = m->next;
        while (l != tail) {
            m->next = l;
            l = m;
            m = r;
            if (r == nullptr) break;
            r = r->next;
        } 
    }
    ListNode* reverseKGroup(ListNode* head, int k) {
        ListNode* dummy = new ListNode(-1);
        dummy->next = head;
        ListNode* pre, *start, *end, *nxt;
        pre = dummy;
        start = pre->next;
        while (start != nullptr) {
            end = pre;
            for (int i = 0; i < k and end != nullptr; ++i) {
                end = end->next;
            }
            if (end == nullptr) break;
            nxt = end->next;
            reverse(start, end);
            pre->next = end;	// 注意这部分代码
            start->next = nxt;	//
            pre = start; //
            start = nxt; //
        }
        return dummy->next;
    }
};

300.最长递增子序列

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
class Solution {
public:
    int lengthOfLIS(vector<int>& nums) {
        int n = (int) nums.size();
        const int inf = 0x3f3f3f3f;
        vector<int> dp((size_t) n, inf);	//注意初始化
        for (int i = 0; i < n; ++i) {
            *lower_bound(dp.begin(), dp.end(), nums[i]) = nums[i];
        }
        return lower_bound(dp.begin(), dp.end(), inf) - dp.begin();
    }
};

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

题意

题解

通过快排,找第n-k个最小元素,快排的时候有个pivot,每次看pivot的位置(全局)跟n - k比较,然后递归查找

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
class Solution {
public:
    int findKthSmallest(vector<int>&nums, int l, int r, int k) {
        int ptrl = l, ptrr = r, x = nums[l];
        if (l < r) {
            while (ptrl < ptrr) {
                while (nums[ptrl] <= x and ptrl < r) ptrl++;
                while (nums[ptrr] >= x and ptrr > l) ptrr--;
                if (ptrl < ptrr) swap(nums[ptrl], nums[ptrr]);
            }
            swap(nums[ptrr], nums[l]);
            if (ptrr == k) return nums[ptrr];	// 注意这里的k是对整个数组而言
            if (ptrr > k) return findKthSmallest(nums, l, ptrr - 1, k);
            else return findKthSmallest(nums, ptrr + 1, r, k);
        }
        return nums[k];	// 注意这里返回nums[k],而不是随意返回
    }
    int findKthLargest(vector<int>& nums, int k) {
        mt19937 rng{random_device{}()};
        shuffle(nums.begin(), nums.end(), rng);
        int n = (int) nums.size();
        return findKthSmallest(nums, 0, n - 1, n - k);	// 注意n - k不是k
    }
};

35.搜索插入位置

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
class Solution {
public:
    int searchInsert(vector<int>& nums, int target) {
        int n = nums.size();
        int l = 0, r = n - 1;
        while (l <= r) {
            int mid = (l + r) / 2;
            if (nums[mid] == target) return mid;
            if (nums[mid] > target) r = mid - 1;
            else l = mid + 1; 
        }
        return l;
    }
};

20.有效的括号

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
class Solution {
public:
    bool isValid(string ss) {
        stack<char> st;
        for (char i : ss) {
            if (i == '(' or i == '[' or i == '{') st.push(i);
            else if (i == ')') {
                if (st.empty() or st.top() != '(') {
                    return false;
                } else {
                    st.pop();
                }
            } 
            else if (i == ']') {
                if (st.empty() or st.top() != '[') {
                    return false;
                } else {
                    st.pop();
                }
            } else {
                if (st.empty() or st.top() != '{') {
                    return false;
                } else {
                    st.pop();
                }
            }
        }
        return st.empty();	// 注意这里
    }
};

121.买卖股票的最佳时机

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
class Solution {
public:
    int maxProfit(vector<int>& prices) {
        int n = prices.size();
        vector<int> sufmax(n);
        sufmax[n - 1] = prices[n - 1];
        for (int i = n - 2; i >= 0; i--) {
            sufmax[i] = max(sufmax[i + 1], prices[i]);
        }
        int ans = 0;
        for (int i = 0; i < n - 1; ++i) {
            ans = max(ans, sufmax[i + 1] - prices[i]);
        }
        return ans;
    }
};

322.零钱兑换

题意

题解

完全背包

对于背包问题的至多、至少、恰好容量,求价值的最大最小值,代码都几乎一致,区别在于初始化

参考

01背包和完全背包,一维二维

初始化的区别

👆这个链接的文章总结得非常好,但是不适用于多重背包问题,如果是多重背包还是老老实实开3个循环,其中一个遍历物品的数量。初始化和求法和01背包、多重背包一致

01背包和完全背包的两种目标函数就是求最大最小价值和求方案数

初始化的规律就是体积至多,求价值最大值时,全初始化为0

其他情况都是dp[0][0] = 0,其他初始化成最大或最小值

体积至多,求方案数,就是dp[0][i] = 1,其他全为0

其他都是dp[0][0] = 1,其他全是0

注意:物品的下标为1

注意:在求至少体积时,状态转移方程稍有差别

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
class Solution {
public:
    int coinChange(vector<int>& coins, int amount) {
        int n = coins.size();
        const int MAX = amount + 3;
        vector<int> dp(MAX, MAX);
        dp[0] = 0;
        for (int i = 0; i < n; ++i) {
            for (int j = coins[i]; j <= amount; ++j) {
                dp[j] = min(dp[j], dp[j - coins[i]] + 1);
            }
        }
        return dp[amount] == MAX ? -1 : dp[amount];
    }
};

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

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
class Solution {
public:
    vector<int> findAnagrams(string s, string p) {
        unordered_map<char, int> mp, mpp;
        for (char i = 'a'; i <= 'z'; i++) {	//预热
            mp[i] = 0;
            mpp[i] = 0;
        }
        vector<int> ans;
        int ns = s.length();
        int np = p.length();
        for (int i = 0; i < np; ++i) {
            mp[s[i]]++;
        }
        for (char i : p) {
            mpp[i]++;
        } 
        int ind = np;
        while (ind <= ns) {
            bool ok = true;
            for (char i = 'a'; i <= 'z'; i++) {
                if (mp[i] != mpp[i]) {
                    ok = false;
                    break;
                }
            }
            if (ok) ans.push_back(ind - np);
            mp[s[ind]]++;
            mp[s[ind - np]]--;
            ind++;
        }
        return ans;
    }
};

239.滑动窗口最大值

题意

题解

用deque维护一个单调序列,比如头部是最大值

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
class Solution {
public:
    vector<int> maxSlidingWindow(vector<int>& nums, int k) {
        deque<int> dq;
        int n = nums.size(); 
        vector<int> ans;
        for (int i = 0; i < n; ++i) {
            if (!dq.empty() and dq.front() <= i - k) {
                dq.pop_front();
            }
            while (!dq.empty() and nums[dq.back()] <= nums[i]) {
                dq.pop_back();
            }
            dq.push_back(i);
            if (i >= k - 1) {
                ans.push_back(nums[dq.front()]);
            }
        }
        return ans;
    }
};

56.合并区间

题意

题解

排序,然后顺序判断前后两个区间能不能合并

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
class Solution {
public:
    vector<vector<int>> merge(vector<vector<int>>& intervals) {
        vector<pair<int, int>> v;
        for (auto& i : intervals) {
            v.emplace_back(i[0], i[1]);
            //cout << i[0] << ' ' << i[1] << '\n';
        }
        sort(v.begin(), v.end());
        vector<pair<int, int>> ans;
        int n = v.size();
        ans.emplace_back(v[0].first, v[0].second);
        for (int i = 1; i < n; i++) {
            auto [l, r] = v[i];
            auto [prel, prer] = ans.back();
            //cout << l << ' ' << r << ' ' << prel << ' ' << prer ;
            if (prer < l) {
                ans.emplace_back(l, r);
            } else {
                ans.pop_back();
                ans.emplace_back(min(l, prel), max(r, prer));
            }
        }
        vector<vector<int>> fans;
        for (auto& [i, j] : ans) {
            fans.push_back(vector<int>{i, j});
        }
        return fans;
    }
};

54.螺旋矩阵

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
class Solution {
public:
    vector<int> spiralOrder(vector<vector<int>>& matrix) {
        vector<int> ans;
        int dir[][2] = {{0, 1}, {1, 0}, {0, -1}, {-1, 0}};
        int now = 0;
        int x = 0, y = 0;
        int n = matrix.size(), m = matrix[0].size();
        bool step = true;
        while ((int) ans.size() < n * m) {
            if (step) {
                ans.push_back(matrix[x][y]);
                matrix[x][y] = -1000;
            }
            int tx = x + dir[now][0], ty = y + dir[now][1];
            if (tx >= 0 and tx < n and ty >= 0 and ty < m and matrix[tx][ty] != -1000) {
                x = tx; y = ty;
                step = true;
            } else {
                now++;
                now %= 4;
                step = false;
            }
        }
        return ans;
    }
};

102.二叉树的层序遍历

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    vector<vector<int>> levelOrder(TreeNode* root) {
        if (root == nullptr) {
            return vector<vector<int>>{};
        }
        queue<pair<TreeNode*, int>> q;
        vector<vector<int>> ans;
        q.emplace(root, 0);
        int now = -1;
        vector<int> tmp;
        bool ok = true;
        while (!q.empty()) {
            auto [node, depth] = q.front(); // 不能写auto& 
            q.pop(); 
            if (depth > now) {
                now = depth;
                if (!ok) ans.push_back(tmp);
                tmp.clear();
                ok = false;
            }
            tmp.push_back(node->val);
            if (node->left != nullptr) {
                q.emplace(node->left, depth + 1);
            }
            if (node->right != nullptr) {
                q.emplace(node->right, depth + 1);
            }
        }
        ans.push_back(tmp);
        return ans;
    }
};

136.只出现一次的数字

题意

题解

异或

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
class Solution {
public:
    int singleNumber(vector<int>& nums) {
        int ans = 0;
        for (int i : nums) {
            ans ^= i;
        }
        return ans;
    }
};

189.轮转数组

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
class Solution {
public:
    void rotate(vector<int>& nums, int k) {
        int n = nums.size();
        k %= n;
        auto reverse = [&](int l, int r) -> void {
            while (l < r) {
                swap(nums[l], nums[r]);
                l++;
                r--;
            }
        };
        reverse(0, n - 1);
        reverse(0, k - 1);
        reverse(k, n - 1);
        return ;
    }
};

48.旋转图像

题意

题解

原地算法,

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
class Solution {
public:
    void rotate(vector<vector<int>>& matrix) {
        int n = matrix.size();
        int nn = n;
        n--;
        for (int i = 0; i < (nn + 1) / 2; i++) {
            for (int j = 0; j < nn / 2; j++) {
                int tmp = matrix[i][j];
                matrix[i][j] = matrix[n - j][i];
                matrix[n - j][i] = matrix[n - i][n - j];
                matrix[n - i][n - j] = matrix[j][n - i];
                matrix[j][n - i] = tmp;
            }
        }
    }
};

141.环形链表

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode(int x) : val(x), next(NULL) {}
 * };
 */
class Solution {
public:
    bool hasCycle(ListNode *head) {
        ListNode* slow, *fast;
        slow = fast = head;
        if (head == nullptr) return false;
        while (true) {
            if (slow->next != nullptr) slow = slow->next; else return false;
            if (fast->next != nullptr) fast = fast->next; else return false;
            if (fast->next != nullptr) fast = fast->next; else return false;
            if (slow == fast) return true;
        }
    }
};

142.环形链表II

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode(int x) : val(x), next(NULL) {}
 * };
 */
class Solution {
public:
    ListNode *detectCycle(ListNode *head) {
        ListNode* slow, *fast;
        slow = fast = head;
        if (head == nullptr) return nullptr;

        while (true) {
            if (slow->next != nullptr) slow = slow->next; else return nullptr;
            if (fast->next != nullptr) fast = fast->next; else return nullptr;
            if (fast->next != nullptr) fast = fast->next; else return nullptr;
            if (slow == fast) {
                ListNode* ptr = head;
                while (ptr != slow) {
                    ptr = ptr->next;
                    slow = slow->next;
                }
                return ptr;
            }
        }
    }
};

98.验证二叉搜索树

题意

题解

中序遍历是有序的

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    bool isValidBST(TreeNode* root) {
        vector<int> v;
        function<void(TreeNode*)> mid = [&](TreeNode* node) {
            if (node == nullptr) return ;
            if (node->left != nullptr) mid(node->left);
            v.push_back(node->val);
            if (node->right != nullptr) mid(node->right);
        };
        mid(root);
        for (int i = 1; i < (int) v.size(); ++i) {
            if (v[i] <= v[i - 1]) return false;
        }
        return true;
    }
};

78.子集

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
class Solution {
public:
    vector<vector<int>> subsets(vector<int>& nums) {
        int n = nums.size();
        vector<vector<int>> ans;
        for (int i = 0; i < (1 << n); i++) {
            vector<int> tmp;
            for (int j = 0; j < n; ++j) {
                if (i >> j & 1) {
                    tmp.push_back(nums[j]);
                }
            }
            ans.push_back(tmp);
        }
        return ans;
    }
};

17.电话号码的字母组合

题意

题解

dfs,dfs(pos)表示当前遍历到pos位置,当前字符串是t

回溯

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
class Solution {
public:
    vector<string> letterCombinations(string digits) {
        int n = digits.size();
        if (n == 0) return vector<string>{};
        unordered_map<char, vector<char>> mp;
        char now = 'a';
        for (char i = '2'; i <= '6'; i++) {
            vector<char> v;
            for (int j = 0; j < 3; ++j) {
                v.push_back(now++);
            }
            mp[i] = v;
        }
        mp['7'] = {'p', 'q', 'r', 's'};
        mp['8'] = {'t', 'u', 'v'};
        mp['9'] = {'w', 'x', 'y', 'z'};
        vector<string> ans;
        function<void(int, string&)> dfs = [&](int pos, string& s) {
            if (pos == n) {
                ans.push_back(s);
                return ;
            }
            for (char i : mp[digits[pos]]) {
                s += i;
                dfs(pos + 1, s);
                s.pop_back();
            }
        };
        string t = "";
        dfs(0, t);
        return ans;
    }
};

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

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
class Solution {
public:
    vector<int> searchRange(vector<int>& nums, int target) {
        int n = nums.size();
        bool got = false;
        auto findPos = [&](int val, bool flag) -> int {
            int l = 0, r = n - 1;
            while (l <= r) {
                int mid = (l + r) / 2;
                if (flag) {
                    if (nums[mid] >= target) {
                        r = mid - 1;
                    } else {
                        l = mid + 1;
                    }
                } else {
                    if (nums[mid] > target) {
                        r = mid - 1;
                    } else {
                        l = mid + 1;
                    }
                }
            }
            if (l >= 0 and l < n and nums[l] == target) got = true;
            return l;
        };
        int res1 = findPos(target, true);
        int res2 = findPos(target + 1, false);
        if (!got) return vector<int>{-1, -1};
        return vector<int>{res1, res2 - 1};
    }
};

41.缺失的第一个正数

题意

题解

遍历数组,只要考虑1~n的数,不断让这个数呆在它对应的位置

如果nums[i] == i + 1就ok了

否则就交换位置,然后不断判断,如果nums[i] == nums[nums[i] - 1]那交换也没用,这个数废了

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
class Solution {
public:
    int firstMissingPositive(vector<int>& nums) {
        int n = nums.size();
        for (int i = 0; i < n; ++i) {
            while (nums[i] > 0 and nums[i] <= n and nums[i] != i + 1 and 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;
    }
};

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

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* removeNthFromEnd(ListNode* head, int n) {
        if (n == 1 and head->next == nullptr) return nullptr;
        ListNode* nxt = head->next;
        ListNode* ptr1, *ptr2;
        ptr1 = head;
        ptr2 = head;
        n++;
        while (n-- and ptr2 != nullptr) {
            // cout << n << '\n';
            // cout << (n and ptr2 != nullptr) << '\n';
            // cout << '\n';
            ptr2 = ptr2->next;
        }
        if (n >= 0) {
            // cout << n ;
            return nxt;
        }
        while (ptr2 != nullptr) {
            ptr1 = ptr1->next;
            ptr2 = ptr2->next;
        }
        ptr1->next = ptr1->next->next;
        return head;
    }
};

114.二叉树展开为链表

题意

题解

看官方题解3

思维

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    void flatten(TreeNode* root) {
        TreeNode* cur = root;
        while (cur != nullptr) {
            if (cur->left != nullptr) {
                TreeNode* pre = cur->left;
                while (pre->right != nullptr) {
                    pre = pre->right;
                }
                pre->right = cur->right;
                cur->right = cur->left;
                cur->left = nullptr;
            }
            cur = cur->right;
        }
    }
};

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

题意

题解

因为值不相同,构建一个哈希表存深度,存父节点,dfs求这些值

求一下深度差,然后让深度深的先走几步,再一起走

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode(int x) : val(x), left(NULL), right(NULL) {}
 * };
 */
class Solution {
public:
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
        unordered_map<int, int> dep;
        unordered_map<int, TreeNode*> par;
        function<void(TreeNode*, int, TreeNode*)> dfs = [&](TreeNode* node, int de, TreeNode* pa) {
            dep[node->val] = de;
            par[node->val] = pa;
            if (node->left != nullptr) {
                dfs(node->left, de + 1, node);
            }
            if (node->right != nullptr) {
                dfs(node->right, de + 1, node);
            }
        };
        dfs(root, 0, nullptr);
        int dif = abs(dep[p->val] - dep[q->val]);
        TreeNode* ll = p, *rr = q;
        if (dep[p->val] > dep[q->val]) {
            while (dif--) {
                ll = par[ll->val];
            }
        } else {
            while (dif--) {
                rr = par[rr->val];
            }
        }
        while (ll != rr) {
            ll = par[ll->val];
            rr = par[rr->val];
        }
        return ll;
    }
};

22.括号生成

题意

题解

这种生成所有方案的题一般用dfs+回溯

dfs(pos, lnum, rnum) 表示当前在pos,左括号有lnum个,右括号有rnum个

dfs退出条件是pos == n * 2

由于要生成有效的括号组合,所以rnum <= lnum

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
class Solution {
public:
    vector<string> generateParenthesis(int n) {
        vector<string> ans;
        string tmp = "";
        function<void(int, int, int)> dfs = [&](int pos, int lnum, int rnum) {
            if (pos == n * 2) {
                ans.push_back(tmp);
                return ;
            }
            if (lnum < n) {
                tmp += '(';
                dfs(pos + 1, lnum + 1, rnum);
                tmp.pop_back();
            } 
            if (rnum < lnum) {
                tmp += ')';
                dfs(pos + 1, lnum, rnum + 1);
                tmp.pop_back();
            }
        };
        dfs(0, 0, 0);
        return ans;
    }
};

33.搜索旋转排序数组

题意

题解

注意各种特判

先找中间值,如果有中间值,两边都二分找一下

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
class Solution {
public:
    int search(vector<int>& nums, int target) {

        int n = nums.size();
        int l = 0, r = n - 1;
        while (l <= r) {
            int mid = (l + r) / 2;
            if (nums[mid] >= nums[0]) {
                l = mid + 1;
            } else {
                r = mid - 1;
            }
        }
        int start = l;	// start是中间位置,就是后半部分的起始
        cout << start << '\n';
        if (start == n) {	//没有中间位置
            l = 0, r = n - 1;
            while (l <= r) {
                int mid = (l + r) / 2;
                if (nums[mid] >= target) {
                    r = mid - 1;
                } else {
                    l = mid + 1;
                }
            }
            cout << l;
            return l < n and nums[l] == target ? l : -1;
        }
        l = 0, r = start - 1;
        while (l <= r) {
            int mid = (l + r) / 2;
            if (nums[mid] >= target) {
                r = mid - 1;
            } else {
                l = mid + 1;
            }
        }
        if (l < start and nums[l] == target) return l;
        l = start, r = n - 1;
        while (l <= r) {
            int mid = (l + r) / 2;
            if (nums[mid] >= target) {
                r = mid - 1;
            } else {
                l = mid + 1;
            }
        }
        if (l < n and nums[l] == target) return l;
        return -1;

    }
};

739.每日温度

题意

题解

思维

单调栈,从前往后遍历,存递减序列

当前数>栈顶(最新的数),那ans[栈顶] = 当前数,然后栈顶没作用了可以滚了(因为它已经更新完前驱),当前数进场,形成递减序列

注意是递减,不是严格递减

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
class Solution {
public:
    vector<int> dailyTemperatures(vector<int>& temperatures) {
        vector<int> ans;
        int n = temperatures.size();
        ans.resize(n);
        stack<int> s;
        for (int i = 0; i < n; ++i) {
            while (!s.empty() and temperatures[s.top()] < temperatures[i]) {
                ans[s.top()] = i - s.top();
                s.pop();
            }
            s.push(i);
        }   
        return ans;
    }
};

55.跳跃游戏

题意

题解

判断最远可达范围

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
class Solution {
public:
    bool canJump(vector<int>& nums) {
        int reach = 0;
        int n = nums.size();
        for (int i = 0; i < n; ++i) {
            if (i > reach) return false;
            reach = max(reach, nums[i] + i);
            //cout << i << ' ' << reach << '\n';
        }
        return true;
    }
};

45.跳跃游戏II

题意

题解

思维

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
class Solution {
public:
    int jump(vector<int>& nums) {
       int n = nums.size();
       if (n == 1) return 0;
       int step = 0;	// 当前步数
       int maxreach = 0;	//step + 1步可达的最远距离
       int end = 0;	//step步可达的最远距离
       for (int i = 0; i < n; ++i) {
           if (i > end) {
               step++;
               end = maxreach;
           }
           maxreach = max(maxreach, i + nums[i]);
       }
       return step;
    }
};

139.单词拆分

题意

题解

关键题

dp

注意单词长度只有20,可以遍历

dp[i] 表示前i个字母是否能在字典中找到

因为单词只有20个字母,所以遍历末尾的单词,dp[i] |= dp[i - x]

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
class Solution {
public:
    bool wordBreak(string s, vector<string>& wordDict) {
        for (auto& i : wordDict) {
            reverse(i.begin(), i.end());
        } 
        int n = s.length();
        int nn = wordDict.size();
        vector<bool> dp(n);
        unordered_set<string> us;
        for (auto& i : wordDict) {
            us.insert(move(i));
        }
        for (int i = 0; i < n; ++i) {
            string tmp = "";
            for (int j = i; j >= max(0, i - 22); j--) {
                tmp += s[j];
                if (us.find(tmp) != us.end()) {
                    if (j - 1 >= 0) {
                        dp[i] = dp[i] | dp[j - 1];
                    } else if (j == 0) {
                        dp[i] = true;
                    }
                }
            }
        }
        for (int i = 0; i < n; ++i) {
            cout << i << ' ' << dp[i] << '\n';
        }
        return dp[n - 1];
    }
};

62.不同路径

题意

题解

dp[i][j] 表示答案

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
class Solution {
public:
    int uniquePaths(int m, int n) {
        vector<vector<int>> dp(m, vector<int>(n));
        dp[0][0] = 1;
        for (int i = 0; i < m; ++i) {
            for (int j = 0; j < n; ++j) {
                if (i == 0 and j == 0) continue;
                if (i == 0) dp[i][j] += dp[i][j - 1];
                else if (j == 0) dp[i][j] += dp[i - 1][j];
                else dp[i][j] = dp[i - 1][j] + dp[i][j - 1];
            } 
        }
        return dp[m - 1][n - 1];
    }
};
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
class Solution {
public:
    int uniquePaths(int m, int n) {
        long long ans = 1;
        for (int i = 1, j = n; i < m; ++i, j++) {
            ans = ans * j / i;
        }
        return ans;
    }
};

64.最小路径和

题意

题解

关键题

dp[i][j] 表示答案

之前做题的时候有遇到vector超时,数组没超时的现象,要注意

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
class Solution {
public:
    int minPathSum(vector<vector<int>>& grid) {
        int n = grid.size();
        int m = grid[0].size();
        vector<vector<int>> dp(n, vector<int>(m));
        for (auto& v : dp) {
            fill(v.begin(), v.end(), 1e8);
        }
        dp[0][0] = grid[0][0];
        for (int i = 0; i < n; ++i) {
            for (int j = 0; j < m; ++j) {
                if (i == 0 and j == 0) continue;
                if (i == 0) {
                    dp[i][j] = min(dp[i][j], dp[i][j - 1] + grid[i][j]);
                } else if (j == 0) {
                    dp[i][j] = min(dp[i][j], dp[i - 1][j] + grid[i][j]);
                } else {
                    dp[i][j] = min({dp[i][j], dp[i - 1][j] + grid[i][j], dp[i][j - 1] + grid[i][j]});
                }
            }
        }
        return dp[n - 1][m - 1];;
    }
};

199.二叉树的右视图

题意

题解

最暴力的做法,bfs,把每一层拿出来,取最后的节点

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    vector<int> rightSideView(TreeNode* root) {
        if (root == nullptr) return vector<int>{};
        vector<int> ans;
        vector<vector<int>> tmp;
        queue<pair<TreeNode*, int>> q;
        int d = -1;
        q.emplace(root, 0);
        vector<int> qq; // 当前层的临时变量
        while (!q.empty()) {
            auto now = q.front();
            q.pop();
            if (now.second != d) {
                if (!qq.empty()) tmp.push_back(qq);
                d++;
                qq.clear();
            }
            qq.push_back(now.first->val);
            if (now.first->left) q.emplace(now.first->left, now.second + 1);
            if (now.first->right) q.emplace(now.first->right, now.second + 1);
        }
        if (!qq.empty()) tmp.push_back(qq);	//最后再push一次
        for (auto& i : tmp) {
            ans.push_back(i.back());
        }
        return ans;
    }
};

437.路径总和III

题意

题解

dfs、dp,dfs(a, b, c)表示以a为根的树,路径和为b的个数,c为true表示路径包含a,false表示路径不包含a

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    int pathSum(TreeNode* root, int targetSum) {
        function<int(TreeNode*, long long, bool)> dfs = [&](TreeNode* node, long long target, bool includ) {
            if (node == nullptr) return 0;
            if (includ) {
                long long v = node->val;
                return (target == v ? 1 : 0) + dfs(node->left, target - v, true) + dfs(node->right, target - v, true);
            } else {
                int l = dfs(node->left, target, true) + dfs(node->left, target, false);
                int r = dfs(node->right, target, true) + dfs(node->right, target, false);
                return l + r;
            }
        };
        return dfs(root, targetSum, true) + dfs(root, targetSum, false);
    }
};

240.搜索二维矩阵II

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
class Solution {
public:
    bool searchMatrix(vector<vector<int>>& matrix, int target) {
        int n = matrix.size();
        int m = matrix[0].size();
        int x = 0, y = m - 1;
        while (x < n and y >= 0) {
            if (matrix[x][y] == target) return true;
            if (matrix[x][y] < target) x++;
            else y--;
        }
        return false;
    }
};

88. 合并两个有序数组

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
class Solution {
public:
    void merge(vector<int>& nums1, int m, vector<int>& nums2, int n) {
        int ptr = m + n - 1;
        int ptr1 = m - 1;
        int ptr2 = n - 1;
        while (ptr1 >= 0 or ptr2 >= 0) {
            if (ptr1 == -1) {
                nums1[ptr] = nums2[ptr2];
                ptr2--;
            } else if (ptr2 == -1) {
                nums1[ptr] = nums1[ptr1];
                ptr1--;
            } else if (nums1[ptr1] <= nums2[ptr2]) {
                nums1[ptr] = nums2[ptr2];
                ptr2--;
            } else {
                nums1[ptr] = nums1[ptr1];
                ptr1--;
            }
            ptr--;
        }
    }
};

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

题意

题解

dfs(node) 表示以node为起点向下的链中最大和

dfs的过程不断更新ans,ans = max(ans, node, dfs(lnode) + node, dfs(rnode) + node, dfs(lnode) + dfs(rnode) + node)

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    int maxPathSum(TreeNode* root) {
        int ans = 0xc0c0c0c0;
        function<int(TreeNode*)> dfs = [&](TreeNode* node) {
            if (node == nullptr) return 0;
            int l = dfs(node->left);
            int r = dfs(node->right);
            ans = max(ans, max(l, r) + node->val);
            ans = max(ans, node->val);
            ans = max(ans, l + r + node->val);
            return max(node->val, max(l, r) + node->val); // 注意这里要max自身节点
        };
        dfs(root);
        return ans;
    }
};

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

题意

题解

前后缀积

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
class Solution {
public:
    vector<int> productExceptSelf(vector<int>& nums) {
        int n = nums.size();
        vector<int> ans(n, 1);
        int prod = 1;
        for (int i = 0; i < n - 1; ++i) {
            prod *= nums[i];	// prefix mul
            ans[i + 1] *= prod;
        }
        prod = 1;
        for (int i = n - 1; i > 0; --i) {
            prod *= nums[i];	// suf mul
            ans[i - 1] *= prod;
        }
        return ans;
    }
};

994.腐烂的橘子

题意

题解

大思路是bfs

最后判一下是不是没有新鲜橘子了

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
class Solution {
public:
    int orangesRotting(vector<vector<int>>& grid) {
        int n = grid.size();
        int m = grid[0].size();
        int ans = 0;
        int dir[][2] = {{0, 1}, {0, -1}, {-1, 0}, {1, 0}};
        queue<pair<int, int>> q;	//存坏橘子位置和时间
        for (int i = 0; i < n; ++i) {
            for (int j = 0; j < m; ++j) {
                if (grid[i][j] == 2) {
                    q.emplace(i * m + j, 0);
                }
            }
        }
        while (!q.empty()) {
            auto now = q.front();
            q.pop();
            ans = max(ans, now.second);
            int x = now.first / m;
            int y = now.first % m;
            for (int i = 0; i < 4; ++i) {
                int tx = x + dir[i][0];
                int ty = y + dir[i][1];
                if (tx >= 0 and tx < n and ty >= 0 and ty < m and grid[tx][ty] == 1) {
                    grid[tx][ty] = 2;
                    q.emplace(tx * m + ty, now.second + 1);
                }
            }
        }
        bool ok = false;
        for (int i = 0; i < n; ++i) {
            for (int j = 0; j < m; ++j) {
                if (grid[i][j] == 1) {
                    ok = true;
                    break;
                }
            }
        }
        return ok ? -1 : ans;
    }
};

39.组合总和

题意

题解

看数据量应该是dfs暴搜+回溯

dfs(pos, sum)表示当前位置,和为sum

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
class Solution {
public:
    vector<vector<int>> combinationSum(vector<int>& candidates, int target) {
        int n = candidates.size();
        set<vector<int>> ans;	// 不用unordered_set,因为要自己写hash
        function<void(vector<int>&, int, int)> dfs = [&](vector<int>& v, int pos, int sum) {
            if (sum == target) {
                ans.insert(v); // 这之后加return 可以不用set
            }
            if (sum > target) return ;
            if (pos >= n) return ;
            v.push_back(candidates[pos]);
            dfs(v, pos, sum + candidates[pos]);	// 可重复选
            dfs(v, pos + 1, sum + candidates[pos]); //把它删掉 可以不用set
            v.pop_back();
            dfs(v, pos + 1, sum); 
        };
        vector<int> tmp;
        dfs(tmp, 0, 0);
        vector<vector<int>> fans(ans.begin(), ans.end());
        return fans;
    }
};

79.单词搜索

题意

题解

dfs暴搜+回溯

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
class Solution {
public:
    bool exist(vector<vector<char>>& board, string word) {
        int n = board.size();
        int m = board[0].size();
        vector<bool> use(n * m + 3);
        int nn = word.length();
        int dir[][2] = {{0, 1}, {0, -1}, {-1, 0}, {1, 0}};
        bool flag = false;
        function<void(int, int, int)> dfs = [&](int x, int y, int pos) {

            if (flag) return ;

            if (pos == nn - 1) {
                flag = true;
                return ;
            }

            use[x * m + y] = true;

            for (int i = 0; i < 4; ++i) {
                int tx = x + dir[i][0];
                int ty = y + dir[i][1];
                if (tx >= 0 and tx < n and ty >= 0 and ty < m and !use[tx * m + ty] and board[tx][ty] == word[pos + 1]) {
                    dfs(tx, ty, pos + 1);
                }
            }
            
            use[x * m + y] = false;

        };
        for (int i = 0; i < n; ++i) {
            for (int j = 0; j < m; ++j) {
                // fill(use.begin(), use.end(), 0);
                if (board[i][j] == word[0]) dfs(i, j, 0);
            }
        }
        return flag;
    }
};

74.搜索二维矩阵

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
class Solution {
public:
    bool searchMatrix(vector<vector<int>>& matrix, int target) {
        int n = matrix.size();
        int m = matrix[0].size();
        int x = 0, y = m - 1;
        while (x < n and y >= 0) {
            if (matrix[x][y] == target) return true;
            if (matrix[x][y] > target) y--; else x++;
        }
        return false;
    }
};

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

题意

题解

相当于之前那个题目的第一步,找中间元素

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
class Solution {
public:
    int findMin(vector<int>& nums) {
        int n = nums.size();
        int l = 0, r = n - 1;
        while (l <= r) {
            int mid = (l + r) / 2;
            if (nums[mid] >= nums[0]) {
                l = mid + 1;
            } else {
                r = mid - 1;
            }
        }
        if (l == n) return nums[0]; // 没旋转过
        return nums[l];
    }
};

155.最小栈

题意

题解

两个栈同步操作

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
class MinStack {
private:
    stack<long long> s, ms;
public:
    MinStack() {
        ms.push(0x3f3f3f3f3f3f3f3f);
    }
    
    void push(int val) {
        s.push(val);
        ms.push(min(ms.top(), (long long) val));
    }
    
    void pop() {
        s.pop();
        ms.pop();
    }
    
    int top() {
        return s.top();
    }
    
    int getMin() {
        return ms.top();
    }
};

/**
 * Your MinStack object will be instantiated and called as such:
 * MinStack* obj = new MinStack();
 * obj->push(val);
 * obj->pop();
 * int param_3 = obj->top();
 * int param_4 = obj->getMin();
 */

394.字符串解码

题意

题解

像这种左括号右括号的题用stack

主要是stack存数字和当前字符串(记录刚开始写的位置),遇到右括号出栈,开始重复

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
class Solution {
public:
    string decodeString(string s) {
        stack<pair<int, int>> st;	//存的是数字,和当前写入位置,先根据s写一遍,然后不断重复num - 1遍
        string ans = "";
        int pos = 0;
        for (int i = 0; i < s.length(); ++i) {
            if (isdigit(s[i])) {	// 如果是数字,获取完整数字
                string num;
                while (i < s.length()) {
                    if (isdigit(s[i])) {
                        num += s[i];
                        i++;
                    } else break;
                }
                int foo = stoi(num);
                st.emplace(foo, pos);
            } else if (s[i] == '[') {
                continue;
            } else if (s[i] == ']') {	// 开始重复num - 1遍
                auto [x, y] = st.top();
                st.pop();
                string tmp = ans.substr(y);
                for (int j = 0; j < x - 1; ++j) {
                    ans += tmp;
                }
                pos = ans.length();
            } else {	// 是字符,开始写第一遍
                ans += s[i];
                pos++;
            }
        }
        return ans;
    }
};

152.乘积最大子数组

题意

题解

dp

由于数有正有负,所以要用两个dp存最大最小值

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
class Solution {
public:
    int maxProduct(vector<int>& nums) {
        int n = nums.size();
        vector<int> mn(n + 3), mx(n + 3);
        int ans = INT_MIN;
        mn[0] = nums[0];
        mx[0] = nums[0];
        for (int i = 1; i < n; ++i) {
            mn[i] = min({nums[i], mn[i - 1] * nums[i], mx[i - 1] * nums[i]});
            mx[i] = max({nums[i], mx[i - 1] * nums[i], mn[i - 1] * nums[i]});
        }
        ans = max(ans, nums[0]);
        for (int i = 1; i < n; ++i) {
            ans = max(ans, mx[i]);
        }
        return ans;
    }
};

72.编辑距离

题意

题解

dp[i][j]表示word1的前i个字符和word2的前j个字符的答案

注意初始化

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
class Solution {
public:
    int minDistance(string word1, string word2) {
        int n = word1.length();
        int m = word2.length();
        word1.insert(word1.begin(), '#');
        word2.insert(word2.begin(), '#');
        vector<vector<int>> dp(n + 3, vector<int>(m + 3, 100000));
        // 1-indexed
        for (int i = 0; i <= n; ++i) {
            dp[i][0] = i;
        }
        for (int i = 0; i <= m; ++i) {
            dp[0][i] = i;
        }
        for (int i = 1; i <= n; ++i) {
            for (int j = 1; j <= m; ++j) {
                if (word1[i] == word2[j]) {
                    dp[i][j] = min(dp[i][j], dp[i - 1][j - 1]);
                }
                dp[i][j] = min(dp[i][j], dp[i - 1][j] + 1);	// 删除
                dp[i][j] = min(dp[i][j], dp[i][j - 1] + 1);	// 增加
                dp[i][j] = min(dp[i][j], dp[i - 1][j - 1] + 1);	// 替换
            }
        }
        return dp[n][m];
    }
};

1143.最长公共子序列

题意

题解

dp[i][j]表示前i个字符和前j个字符的答案

只要判断s1[i] == s2[j]的情况

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
class Solution {
public:
    int longestCommonSubsequence(string text1, string text2) {
        int n = text1.length();
        int m = text2.length();
        text1.insert(text1.begin(), '$');
        text2.insert(text2.begin(), '3');
        vector<vector<int>> dp(n + 3, vector<int>(m + 3));
        for (int i = 1; i <= n; ++i) {
            for (int j = 1; j <= m; ++j) {
                if (text1[i] == text2[j]) {
                    dp[i][j] = max(dp[i][j], dp[i - 1][j - 1] + 1);
                } else {
                    dp[i][j] = max(dp[i][j], max(dp[i - 1][j], dp[i][j - 1]));
                }
            }
        }
        return dp[n][m];
    }
};

763.划分字母区间

题意

题解

求每个字母最后出现的位置

从左往右遍历

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
class Solution {
public:
    vector<int> partitionLabels(string s) {
        vector<int> ans;
        int n = s.length();
        unordered_map<char, int> mp;
        for (int i = 0; i < n; ++i) {
            mp[s[i]] = i;
        }
        int pos = 0;	// 新子区间的起点
        char now = s[0];	//起点字符
        while (true) {	// 大循环是遍历多个子区间
            int end = mp[now];	// 当前子区间要遍历到的结尾
            int tmp = pos;	// 当前遍历到的位置
            while (true) {	// 小循环是遍历子区间内的元素
                if (mp[s[tmp]] > end) {	// 扩大end
                    end = mp[s[tmp]];
                    tmp++;	// 往后一格
                } else if (tmp < end) {	// 无事发生
                    tmp++;
                } else break;	// 到底了,可以打包这个子区间了
            }
            ans.push_back(end - pos + 1);
            pos = end + 1;	// 下一个子区间的起点
            now = s[pos];
            if (pos >= n) break;
        }
        return ans;
    }
};

131.分割回文传

题意

题解

一看到求方案,想到dfs回溯

预处理s[i][j]是否是回文串,然后dfs,回溯

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
class Solution {
public:
    vector<vector<string>> partition(string s) {
        vector<vector<string>> ans;
        int n = s.length();
        vector<vector<bool>> dp(n + 3, vector<bool>(n + 3));
        for (int i = 1; i <= n; ++i) { // len
            for (int j = 0; j < n - i + 1; ++j) {
                if (i == 1) {
                    dp[j][j] = true;
                } else if (i == 2) {
                    dp[j][j + 1] = (s[j] == s[j + 1]);
                } else {
                    dp[j][j + i - 1] = (s[j] == s[j + i - 1] and dp[j + 1][j + i - 2]);
                }
            }
        }
        vector<string> tmp;
        function<void(int)> dfs = [&](int pos) {
            if (pos == n) {
                ans.emplace_back(tmp);
                return ;
            }
            for (int i = pos; i < n; ++i) {
                if (dp[pos][i]) {
                    tmp.push_back(s.substr(pos, i - pos + 1));
                    dfs(i + 1);
                    tmp.pop_back();
                }
            }
        };
        dfs(0);
        return ans;
    }
};

416.分割等和子集

题意

题解

经典01🎒求方案问题

注意特判边界,注意逻辑

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
class Solution {
public:
    bool canPartition(vector<int>& nums) {
        int sum = accumulate(nums.begin(), nums.end(), 0);
        if (sum & 1) return false;
        sum /= 2;
        int n = nums.size();
        if (n == 1) return false; // !
        vector<vector<bool>> dp(n + 3, vector<bool>(sum + 3));
        if (nums[0] < sum + 3) {    // !
            dp[0][nums[0]] = true;
        }
        for (int i = 1; i < n; ++i) {
            for (int j = 0; j < sum + 3; ++j) {
                dp[i][j] = dp[i][j] or dp[i - 1][j];
                if (j - nums[i] >= 0) {
                    dp[i][j] = dp[i][j] or dp[i - 1][j - nums[i]];
                }
                // cout << i << ' ' << j << ' ' << dp[i][j] << '\n';
            }
        }
        // cout << dp[0][50] << '\n';
        return dp[n - 1][sum];
    }
};

这个代码逻辑更清晰

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
class Solution {
public:
    bool canPartition(vector<int>& nums) {
        int n = nums.size();
        int sum = accumulate(nums.begin(), nums.end(), 0);
        if (sum % 2 != 0) return false;
        sum /= 2;
        vector<vector<bool>> dp(n + 3, vector<bool>(sum + 3));
        dp[0][0] = true;
        nums.insert(nums.begin(), 0);
        for (int i = 1; i <= n; ++i) {
            for (int j = 0; j <= sum; ++j) {
                dp[i][j] = dp[i][j] or dp[i - 1][j];
                if (nums[i] <= j) {
                    dp[i][j] = dp[i][j] or dp[i - 1][j - nums[i]];
                }
            }
        }
        return dp[n][sum];
    }
};

75.颜色分类

题意

题解

计数

思维

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
class Solution {
public:
    void sortColors(vector<int>& nums) {
        int z = 0, o = 0, t = 0;
        int n = nums.size();
        for (int i : nums) {
            if (i == 0) z++; else if (i == 1) o++; else t++;
        }
        for (int i = 0; i < z; ++i) {
            nums[i] = 0;
        }
        for (int i = z; i < z + o; ++i) {
            nums[i] = 1;
        }
        for (int i = z + o; i < n; ++i) {
            nums[i] = 2;
        }
        return ;
    }
};

在草稿纸上模拟一下

p0是下个0至少应该在的位置,p1是下个1至少应该在的位置

p1和p0之间是0和1

当前数如果是1,直接调换,

当前数如果0,调换,如果p0 < p1就会把1调出去,1就在2的后面了,所以要把1跟p1交换

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
class Solution {
public:
    void sortColors(vector<int>& nums) {
        int p0 = 0, p1 = 0;
        int n = nums.size();
        for (int i = 0; i < n; ++i) {
            if (nums[i] == 1) {
                swap(nums[i], nums[p1]);
                p1++;
            } else if (nums[i] == 0) {
                swap(nums[i], nums[p0]);
                if (p0 < p1) {
                    swap(nums[i], nums[p1]);
                }
                p0++;
                p1++;
            }
        }
    }
};

31.下一个排列

题意

题解

思维,注意特判

比如524987663,答案是526346789

首先找最后一个相邻顺序对,就是49,要把4换掉,用4之后比4大的最小的数交换,这里是6,不能用5

然后把4之后的数排序

找“4之后比4大的最小的数”有个技巧就是从后往前找,第一个>4的数就是目标,因为4之后的数都是倒序

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
class Solution {
public:
    void nextPermutation(vector<int>& nums) {
        int n = nums.size();
        // find the last ordered pair which is nums[pos] and nums[pos + 1]
        int pos = -1;
        for (int i = n - 2; i >= 0; --i) {
            if (nums[i] < nums[i + 1]) {
                pos = i;
                break;
            }
        }
        if (pos == -1) {    // corner case
            reverse(nums.begin(), nums.end());
            return ;
        }
        // find the last number greater than nums[pos], then swap
        for (int i = n - 1; i > pos; --i) {
            if (nums[i] > nums[pos]) {
                swap(nums[i], nums[pos]);
                break;
            }
        }
        // make the sequence after pos ordered
        reverse(nums.begin() + pos + 1, nums.end());
    }
};

287.寻找重复数

题意

题解

思维

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
class Solution {
public:
    int findDuplicate(vector<int>& nums) {
        int slow = 0, fast = 0;
        do {
            slow = nums[slow];
            fast = nums[nums[fast]];
        } while (slow != fast);
        slow = 0;
        while (slow != fast) {
            slow = nums[slow];
            fast = nums[fast];
        }
        return slow;
    }
};

51.N皇后

题意

题解

方案题,dfs+回溯

由于条件限制比较多,要用很多数据结构来存储临时数据

dfs(row)表示处理第row行

一次dfs是处理一行,遍历这行所有的列,判断是否可以落子

用三个集合存储当前哪些列,哪些正对角线,哪些反对角线有棋子

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
class Solution {
public:
    vector<vector<string>> solveNQueens(int n) {
        vector<vector<string>> ans;
        unordered_set<int> col, diag1, diag2;
      	// col存当前哪些列有皇后
      	// diag存当前对角线上有没皇后,对角线上r - c和r + c值不变
        vector<int> queen(n, -1);	//第i行的皇后在哪里列
        auto genBoard = [&]() -> vector<string> {
            vector<string> ret;
            for (int i = 0; i < n; ++i) {
                string tmp = string(n, '.');
                tmp[queen[i]] = 'Q';
                ret.push_back(tmp);
            }
            return ret;
        };
        function<void(int)> dfs = [&](int row) {
            if (row == n) {
                auto board = genBoard();
                ans.push_back(board);
                return ;
            }
            for (int c = 0; c < n; ++c) {	// 遍历每一列
                if (col.find(c) != col.end()) continue;
                if (diag1.find(row - c) != diag1.end()) continue;
                if (diag2.find(row + c) != diag2.end()) continue;
              
                queen[row] = c;
                col.insert(c);
                diag1.insert(row - c);
                diag2.insert(row + c);
              
                dfs(row + 1);
              
                // backtrace
                queen[row] = -c;
                col.erase(c);
                diag1.erase(row - c);
                diag2.erase(row + c);
            }
        };
        dfs(0);
        return ans;
    }
};

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

题意

题解

比较巧妙的二分

细节非常多,很容易写bug,调了半天

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
class Solution {
public:
    double findMedianSortedArrays(vector<int>& nums1, vector<int>& nums2) {
        int nm = nums1.size() + nums2.size();
        // findk表示从v1的pos1开始,从v2的pos2开始,找这两者中第k小的数
        function<int(vector<int>&, int, vector<int>&, int, int)> findk = [&](vector<int>& v1, int pos1, vector<int>& v2, int pos2, int k) {
            if (v1.size() - pos1 > v2.size() - pos2) {  // 这句话写在函数的最前面
                return findk(v2, pos2, v1, pos1, k);    // 始终保持v1长度不大于v2
            }
            if (pos1 == v1.size()) {    // 由于v1的长度不大于v2,如果v1已经搜到头了,直接返回v2的第k个元素
                return v2[pos2 + k - 1];
            }
            if (k == 1) return min(v1[pos1], v2[pos2]); // 当k=1时,没法分给两个数组
            int num = min(k / 2, (int) v1.size() - pos1);   // v1取第num个,v2取第k - num个
            if (v1[pos1 + num - 1] < v2[pos2 + k - num - 1]) {  // 如果v1[num] < v2[k - num]那就说明第k个数不可能在v1的前num个,所以排除
                return findk(v1, pos1 + num, v2, pos2, k - num);
            } else {
                return findk(v1, pos1, v2, pos2 + k - num, num);
            }
        };
        if (nm % 2 == 0) {
            int l = findk(nums1, 0, nums2, 0, nm / 2);
            int r = findk(nums1, 0, nums2, 0, nm / 2 + 1);
            return (l + r) / 2.0;
        } else {
            return findk(nums1, 0, nums2, 0, nm / 2 + 1);
        }
    }
};

84.柱状图中最大的矩形

题意

题解

思维,单调栈

什么时候出栈,当这个元素用于更新ans,没有作用了就出栈

看官方视频题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
class Solution {
public:
    int largestRectangleArea(vector<int>& heights) {
        heights.push_back(0); // -1 is also ok 注意一下这行代码
        stack<int> s;
        s.push(-1);
        int ans = 0;
        for (int i = 0; i < heights.size(); ++i) {
            while (s.size() > 1 and heights[s.top()] > heights[i]) {
                auto tmp = s.top();
                s.pop();
                ans = max(ans, (i - s.top() - 1) * heights[tmp]);
            }
            s.push(i);
        }
        return ans;
    }
};

32.最长有效括号

题意

题解

遍历的时候记录左括号和右括号的数量,如果相等就更新答案,右>左就全置零

倒着再来一遍,以防((())

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
class Solution {
public:
    int longestValidParentheses(string s) {
        int left = 0, right = 0;
        int ans = 0;
        for (char i : s) {
            if (i == '(') left++; else right++;
            if (left == right) ans = max(ans, left * 2);
            if (right > left) left = right = 0;
        }
        reverse(s.begin(), s.end());
        left = right = 0;
        for (char i : s) {
            if (i == ')') left++; else right++;
            if (left == right) ans = max(ans, left * 2);
            if (right > left) left = right = 0;
        }
        return ans;
    }
};

118.杨辉三角

题意

题解

模拟

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
class Solution {
public:
    vector<vector<int>> generate(int numRows) {
        vector<vector<int>> ans;
        ans.push_back(vector<int>{1});
        while ((int) ans.size() < numRows) {
            auto tmp = ans.back();
            vector<int> v;
            v.push_back(1);
            for (int i = 0; i < (int) tmp.size() - 1; i++) {
                v.push_back(tmp[i] + tmp[i + 1]);
            }
            v.push_back(1);
            ans.push_back(v);
        }
        return ans;
    }
};

70.爬楼梯

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
class Solution {
public:
    int climbStairs(int n) {
        int p, q, r;
        p = q = r = 1;
        for (int i = 2; i <= n; ++i) {
            p = q + r;
            r = q;
            q = p;
        }
        return p;
    }
};
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
class Solution {
public:
    using mat = vector<vector<long long>>;
    mat matmul(mat& A, mat& B) {
        mat C = mat(A.size(), vector<long long>(B[0].size()));
        for (int i = 0; i < A.size(); ++i) {
            for (int k = 0; k < B.size(); ++k) {
                for (int j = 0; j < B[0].size(); ++j) {
                    C[i][j] += A[i][k] * B[k][j];
                }
            }
        }
        return C;
    }
    mat quickPow(mat& A, int n) {
        mat ret = A;
        for (int i = 0; i < A.size(); ++i) {
            for (int j = 0; j < A[0].size(); ++j) {
                ret[i][j] = 1;
            }
        }
        while (n > 0) {
            if (n & 1) {
                ret = matmul(ret, A);
            }
            A = matmul(A, A);
            n >>= 1;
        }
        return ret;
    }
    int climbStairs(int n) {
        if (n == 1) return 1; 
        mat core = {{1, 1}, {1, 0}};
        mat ans = quickPow(core, n - 2);
        return ans[1][0] + ans[1][1];
    }
};

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

题意

题解

dfs 构建🌲

build(x, y)表示构建区间x~y的平衡二叉树

因为已经有序,根节点是中间元素

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    TreeNode* sortedArrayToBST(vector<int>& nums) {
        function<TreeNode*(int, int)> build = [&](int l, int r) -> TreeNode* {	// 要加-> TreeNode*
            if (l > r) return nullptr;
            int mid = (l + r) / 2;
            TreeNode* node = new TreeNode(nums[mid]);
            node->left = build(l, mid - 1);
            node->right = build(mid + 1, r);
            return node;
        };
        return build(0, (int) nums.size() - 1);
    }
};

543.二叉树的直径

题意

题解

算深度

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    int diameterOfBinaryTree(TreeNode* root) {
        int ans = 0;   
        function<int(TreeNode*)> dfs = [&](TreeNode* node) {    // 从node出发向下的最大深度
            int l = -1, r = -1;
            if (node->left != nullptr) {
                l = dfs(node->left);
            }
            if (node->right != nullptr) {
                r = dfs(node->right);
            }
            ans = max(ans, l + r + 2);
            return max(l, r) + 1;
        };
        dfs(root);
        return ans;
    }
};

101.对称二叉树

题意

题解

递归

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    bool isSymmetric(TreeNode* root) {
        function<bool(TreeNode*, TreeNode*)> same = [&](TreeNode* node1, TreeNode* node2) {
            if (node1 != nullptr and node2 != nullptr) {
                return (node1->val == node2->val) && same(node1->left, node2->right) && same(node1->right, node2->left);
            } else if (node1 == nullptr and node2 == nullptr) return true;
            return false;
        };
        return same(root->left, root->right);
    }
};

迭代

用队列存pair,左节点和右节点

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    bool isSymmetric(TreeNode* root) {
        queue<pair<TreeNode*, TreeNode*>> q;
        q.emplace(root->left, root->right);
        while (!q.empty()) {
            auto [l, r] = q.front();
            q.pop();
            if (l == nullptr and r == nullptr) continue;
            if (l != nullptr and r != nullptr) {
                if (l->val != r->val) return false;
                q.emplace(l->left, r->right);
                q.emplace(l->right, r->left);
            } else {
                return false;
            }
        }
        return true;
    }
};

226.翻转二叉树

题意

题解

dfs

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    TreeNode* invertTree(TreeNode* root) {
        if (root == nullptr) return nullptr;
        auto l = invertTree(root->right);
        auto r = invertTree(root->left);
        root->left = l;
        root->right = r;
        return root;
    }
};

104.二叉树的最大深度

题意

题解

dfs

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    int maxDepth(TreeNode* root) {
        if (root == nullptr) return 0;
        return max(maxDepth(root->left), maxDepth(root->right)) + 1;
    }
};

bfs

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    int maxDepth(TreeNode* root) {
        if (root == nullptr) return 0;
        queue<pair<TreeNode*, int>> q;
        q.emplace(root, 0);
        int ans = 0;
        while (!q.empty()) {
            auto tmp = q.front();
            q.pop();
            ans = max(ans, tmp.second);
            if (tmp.first->left != nullptr) q.emplace(tmp.first->left, tmp.second + 1);
            if (tmp.first->right != nullptr) q.emplace(tmp.first->right, tmp.second + 1);
        }
        return ans + 1;
    }
};

206.反转链表

题意

题解

迭代

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* reverseList(ListNode* head) {
        ListNode* l, *mid, *r;
        l = nullptr, mid = head;
        while (mid) {
            r = mid->next;
            mid->next = l;
            l = mid;
            mid = r;
        }
        return l;
    }
};

递归(很抽象)

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* reverseList(ListNode* head) {
        if (head == nullptr or head->next == nullptr) {
            return head;
        }
        ListNode* newHead = reverseList(head->next);
        head->next->next = head;
        head->next = nullptr;
        return newHead;
    }
};

234.回文链表

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    bool isPalindrome(ListNode* head) {
        if (head->next == nullptr) return true; // corner case, size = 1, cannot separate
        if (head->next->next == nullptr) return head->val == head->next->val; // corner case, size = 2, cannot seperate
        ListNode* slow, *fast;
        slow = fast = head;
        bool odd;
        while (true) {
            slow = slow->next;  // notice the order
            if (!fast->next) {
                odd = true;
                break;
            }
            if (!fast->next->next) {
                odd = false;
                break;
            }
            fast = fast->next->next;
        }
        ListNode* r = slow, *l = head;
        while (odd ? l->next->next != r : l->next != r) {
            l = l->next;
        }
        l->next = nullptr;
        // reverse
        ListNode* prev, *mid, *post;
        prev = nullptr, mid = head;
        while (mid) {
            post = mid->next;
            mid->next = prev;
            prev = mid;
            mid = post;
        }
        while (l != nullptr) {
            if (l->val != r->val) return false;
            l = l->next;
            r = r->next;
        }
        return true;
    }
};

11.盛最多水的容器

题意

题解

双指针

容量取决于两根柱子的距离和最短的柱子,可以由公式计算

双指针表示这两根柱子,从最左和最右往中间逼近

如果height[i] < height[j] 那么i++,这样才有机会寻找更大的容量

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
class Solution {
public:
    int maxArea(vector<int>& height) {
        int n = height.size();
        int l = 0, r = n - 1;
        int ans = 0;
        while (l < r) {
            ans = max(ans, min(height[l], height[r]) * (r - l));
            if (height[l] < height[r]) {
                l++;
            } else {
                r--;
            }
        }
        return ans;
    }
};

15.三数之和

题意

题解

分情况讨论,两个数相同和三个数都不相同

用哈希表计数,对原数组去重

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        sort(nums.begin(), nums.end());
        unordered_map<int, int> mp;	//个数
        for (int i : nums) {
            mp[i]++;
        }
        nums.erase(unique(nums.begin(), nums.end()), nums.end()); 
        vector<vector<int>> ans;
        int n = nums.size();
        for (int i = 0; i < n; ++i) {
            for (int j = i + 1; j < n; ++j) {
                mp[nums[i]]--;
                mp[nums[j]]--;
                int now = -nums[i] - nums[j];
              // 为了不重复计算,假设三数按顺序排
                if (now > nums[j] and mp.find(now) != mp.end() and mp[now] > 0) {
                    ans.push_back({nums[i], nums[j], now});
                }
                mp[nums[i]]++;
                mp[nums[j]]++;
            }
        }
        for (int i = 0; i < n; ++i) {
            if (mp[nums[i]] > 1) {
                mp[nums[i]] -= 2;
                int now = -nums[i] - nums[i];
                if (mp.find(now) != mp.end() and mp[now] > 0) {
                    ans.push_back({nums[i], nums[i], now});
                }
                mp[nums[i]] += 2;
            }
        }
        return ans;
    }
};

42.接雨水

题意

题解

和盛最多水的容器有点像

双指针,前后缀

对于每一列,水的容量是min(左边柱子最大高度 - 右边柱子最大高度) - 当前柱子高度

用双指针一左一右往中间逼近,不断求指针那一列的水的容量

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
class Solution {
public:
    int trap(vector<int>& height) {
        int n = height.size();
        int l = 0, r = n - 1;
        int ans = 0;
        int premax = 0, sufmax = 0;
        while (l < r) { // l <= r is also ok, l and r will meet in the highest block which volume is 0
            premax = max(premax, height[l]);
            sufmax = max(sufmax, height[r]);
            if (premax < sufmax) {
                ans += premax - height[l];
                l++;
            } else {
                ans += sufmax - height[r];
                r--;
            }
        }
        return ans;
    }
};

76.最小覆盖子串

题意

题解

滑动窗口

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
class Solution {
public:
    string minWindow(string s, string t) {
        int n = s.length();
        unordered_map<char, int> mp, mpp;	// mp对t计数,mpp对滑窗计数
        for (char i : t) {
            mp[i]++;
        }
        int l = 0, r = 0;
        mpp[s[0]] = 1;
        int ans = 0x3f3f3f3f;
        int pos = -1;
        while (l <= r) {
            bool ok = true;
            for (auto& [x, y] : mp) {
                if (mpp.find(x) != mpp.end() and mpp[x] >= y) ;
                else {
                    ok = false;
                    break;
                }
            }
            if (ok) {
                if (r - l + 1 < ans) {
                    ans = r - l + 1;
                    pos = l;
                }
                mpp[s[l]]--;
                l++;
            } else if (r == n) {
                mpp[s[l]]--;
                l++;
            } else {
                r++;
                if (r < n) mpp[s[r]]++;
            }
        }
        return pos == -1 ? "" : s.substr(pos, ans);
    }
};

21.合并两个有序链表

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {
        ListNode* dummy = new ListNode;
        ListNode* prev = dummy;
        ListNode* ptr1 = list1, *ptr2 = list2;
        while (ptr1 and ptr2) {
            if (ptr1->val <= ptr2->val) {
                prev->next = ptr1;
                prev = ptr1;
                ptr1 = ptr1->next;
            } else {
                prev->next = ptr2;
                prev = ptr2;
                ptr2 = ptr2->next;
            }
        }
        prev->next = (ptr1 ? ptr1 : ptr2);
        return dummy->next;
    }
};

2.两数相加

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) {
        ListNode* ptr1 = l1, *ptr2 = l2;
        bool carry = false;
        ListNode* dummy = new ListNode;
        ListNode* ptr = dummy;
        while (ptr1 and ptr2) {
            int v = ptr1->val + ptr2->val + carry;
            if (v > 9) {
                carry = true;
                v -= 10;
            } else carry = false;
            ptr->next = new ListNode(v);
            ptr = ptr->next;
            ptr1 = ptr1->next;
            ptr2 = ptr2->next;
        }
        while (ptr1) {
            int v = ptr1->val + carry;
            if (v > 9) {
                carry = true;
                v -= 10;
            } else carry = false;
            ptr->next = new ListNode(v);
            ptr = ptr->next;
            ptr1 = ptr1->next;
        }
        while (ptr2) {
            int v = ptr2->val + carry;
            if (v > 9) {
                carry = true;
                v -= 10;
            } else carry = false;
            ptr->next = new ListNode(v);
            ptr = ptr->next;
            ptr2 = ptr2->next;
        }
        if (carry) {
            ptr->next = new ListNode(1);
        }
        return dummy->next;
    }
};

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

题意

题解

递归

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* swapPairs(ListNode* head) {
        if (!head or !head->next) return head;
        ListNode* newHead = head->next;
        head->next = swapPairs(newHead->next);
        newHead->next = head;
        return newHead;
    }
};

迭代,题解参看官方题解的图解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* swapPairs(ListNode* head) {
        ListNode* dummy = new ListNode;
        dummy->next = head;
        ListNode* ptr1 = dummy;
        ListNode* ptr2, *ptr3;
        while (ptr1->next and ptr1->next->next) {
            ptr2 = ptr1->next;
            ptr3 = ptr2->next;
            ptr1->next = ptr3;
            ptr2->next = ptr3->next;
            ptr3->next = ptr2;
            ptr1 = ptr2;
        }
        auto ans = dummy->next;
        delete dummy;
        return ans;
    }
};

138.随机链表的复制

题意

题解

在每个节点后面构造替身

A->A’->B->B’->C->C'

根据A、B、C的random指针,让A’ B’ C’ 的random指针指向对应的替身

断掉A和A’和B的连接

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
/*
// Definition for a Node.
class Node {
public:
    int val;
    Node* next;
    Node* random;
    
    Node(int _val) {
        val = _val;
        next = NULL;
        random = NULL;
    }
};
*/

class Solution {
public:
    Node* copyRandomList(Node* head) {
        if (!head) return nullptr;  // !!
        Node* ptr = head;
        // A->A'->B->B'->C->C'
        while (ptr) {
            Node* tmp = new Node(ptr->val);	// 构造替身
            tmp->next = ptr->next;	// 连接后者
            ptr->next = tmp;	//连接前者
            ptr = ptr->next->next;	//下一个
        }
        // make random pointer of new list correct
        ptr = head;
        while (ptr) {
            if (ptr->random) ptr->next->random = ptr->random->next; // !!
            ptr = ptr->next->next;
        }
        auto ans = head->next;
        // set both next pointer back
        ptr = head;
        while (ptr) {
            auto nxt = ptr->next;
            if (ptr->next) ptr->next = ptr->next->next; // !!
            ptr = nxt;
        }
        return ans;
    }
};

23.合并K个升序链表

题意

题解

时间复杂度O(knlogk)

空间复杂度O(k)

此处我用的是最直接的方法,把lists中的第一个元素加到小顶堆中,然后取出元素,添加它的next

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    struct cmp {
        bool operator() (const ListNode* node1, const ListNode* node2) const {  // don't use &
            return node1->val > node2->val;
        }
    };
    ListNode* mergeKLists(vector<ListNode*>& lists) {
        priority_queue<ListNode*, vector<ListNode*>, cmp> pq;   // vector<ListNode*>
        ListNode* ans = new ListNode(0);
        ListNode* ptr = ans;
        for (ListNode* i : lists) {
            if (i) pq.emplace(i);
        } 
        while (!pq.empty()) {
            auto tmp = pq.top();
            pq.pop();
            ptr->next = tmp;
            if (tmp->next) pq.emplace(tmp->next);
            ptr = ptr->next;
        }
        return ans->next;
    }
};

146.LRU缓存

题意

题解

哈希表+双向链表

哈希表存key和节点的映射

每个节点都有key和value

双向链表的头部是最新的数据,尾部是最旧的数据,每次get(k)时,把节点调到最前面

put(k,v)时先找有没有,有的话就更新值,调到最前面,没有的话生成节点直接插到最前面,如果超出容量,把最后一个节点删掉,记得在哈希表中删对应的key

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
class LRUCache {
private:
    struct DListNode {
        int key, val;
        DListNode* prev, *next;
        DListNode() : key(0), val(0), prev(nullptr), next(nullptr) {};
        DListNode(int k, int v) : key(k), val(v), prev(nullptr), next(nullptr) {}
    };
    DListNode* head, *tail;
    unordered_map<int, DListNode*> mp;
    size_t sz, cap;

    void disconnect(DListNode* node) {
        node->prev->next = node->next;
        node->next->prev = node->prev;
    }

    void insertHead(DListNode* node) {
        node->next = head->next;
        head->next->prev = node;
        head->next = node;
        node->prev = head;
    }

    void evictNode() {
        DListNode* node = tail->prev;
        node->prev->next = node->next;
        node->next->prev = node->prev;
        mp.erase(node->key); // !
        delete node;
    }
 
public:
    LRUCache(int capacity) : sz(0), cap(capacity) {
        head = new DListNode();
        tail = new DListNode();
        head->next = tail;
        tail->prev = head;
    }
    
    int get(int key) {
        if (mp.find(key) != mp.end()) {
            auto node = mp[key];
            disconnect(node);
            insertHead(node);
            return node->val;
        } else return -1;
    }
    
    void put(int key, int value) {
        if (mp.find(key) != mp.end()) {
            auto node = mp[key];
            node->val = value;
            disconnect(node);
            insertHead(node);
        } else {
            DListNode* node = new DListNode(key, value);
            mp[key] = node;
            insertHead(node);
            sz++;
            if (sz > cap) {
                sz--;
                evictNode();
            }
        }
    }
};

/**
 * Your LRUCache object will be instantiated and called as such:
 * LRUCache* obj = new LRUCache(capacity);
 * int param_1 = obj->get(key);
 * obj->put(key,value);
 */

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

题意

题解

简单版是二叉树的中序遍历是有序的,时间复杂度是O(n)

进阶版是针对频繁增删

用哈希表存节点为根的树的节点个数,但是我不知道这种方法有什么意义

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    int kthSmallest(TreeNode* root, int k) {
        unordered_map<TreeNode*, int> mp;
        function<int(TreeNode*)> dfs = [&](TreeNode* node) {
            if (node == nullptr) return 0;
            int ret = 1 + dfs(node->left) + dfs(node->right);
            mp[node] = ret;
            return ret;
        };
        dfs(root);
        function<int(TreeNode*, int)> find = [&](TreeNode* node, int foo) {
            int l = 0, r = 0;
            if (node->left) l = mp[node->left];
            if (node->right) r = mp[node->right];
            if (foo <= l) {
                return find(node->left, foo);
            } else if (l + 1 == foo) {
                return node->val;
            } else {
                return find(node->right, foo - l - 1);
            }
        };
        return find(root, k);
    }
};

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

题意

题解

递归,找根

用哈希表存元素位置

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
private:
    unordered_map<int, int> mp;
public:
    TreeNode* sol(vector<int>& pre, vector<int>& in, int prel, int prer, int inl, int inr) {
        if (prel > prer) return nullptr; // !!
        int rt = pre[prel];
        TreeNode* node = new TreeNode(rt);
        int leftSize = mp[rt] - inl;
        node->left = sol(pre, in, prel + 1, prel + leftSize, inl, inl + leftSize - 1);
        node->right = sol(pre, in, prel + leftSize + 1, prer, inl + leftSize + 1, inr);
        return node;
    }
    TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder) {
        int n = preorder.size();
        for (int i = 0; i < n; ++i) {
            mp[inorder[i]] = i;
        }
        return sol(preorder, inorder, 0, n - 1, 0, n - 1);
    }
};

迭代

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    struct info {
        TreeNode* node;
        int prel, prer, inl, inr;
    };
    TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder) {
        int n = preorder.size();
        stack<info> s;
        unordered_map<int, int> mp;
        for (int i = 0; i < n; ++i) {
            mp[inorder[i]] = i;
        }
        TreeNode* ans = new TreeNode(preorder[0]);
        if (n == 1) return ans;
        s.emplace(ans, 0, n - 1, 0, n - 1);
        while (!s.empty()) {
            auto now = s.top();
            s.pop();
            auto node = now.node;
            int prel = now.prel, prer = now.prer, inl = now.inl, inr = now.inr;
            // if (prel > prer) {
            //     // node->left = nullptr;
            //     // node->right = nullptr;
            //     continue;
            // }
            int rt = node->val;
            int leftsize = mp[rt] - inl;
            cout << prel;
            if (1 <= leftsize) {
                TreeNode* lnode = new TreeNode(preorder[prel + 1]);
                node->left = lnode;
                s.emplace(lnode, prel + 1, prel + leftsize, inl, inl + leftsize - 1);
            }
            if (prel + leftsize + 1 <= prer) {
                TreeNode* rnode = new TreeNode(preorder[prel + leftsize + 1]);
                node->right = rnode;
                s.emplace(rnode, prel + leftsize + 1, prer, inl + leftsize + 1, inr);
            }
        }
        return ans;
    }
};

279.完全平方数

题意

题解

思维

有数学公式,如果一个数可以表示成4 ^ k * (8 * m + 7),就是4,否则一定小于4,小于4的分情况讨论

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
class Solution {
public:
    int numSquares(int n) {
        auto isPerfectSquare = [&](int x) -> bool {
            int y = sqrt(x);
            return y * y == x;
        };
        if (isPerfectSquare(n)) {
            return 1;
        }
        for (int i = 1; i * i < n; ++i) {
            if (isPerfectSquare(n - i * i)) return 2;
        }
        // 4 ^ k * (8 * m + 7)
        while (n % 4 == 0) {
            n /= 4;
        }
        if (n % 8 == 7) return 4;
        return 3;
    }
};

198.打家劫舍

题意

题解

dp[i]表示前i个数的答案

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
class Solution {
public:
    int rob(vector<int>& nums) {
        int n = nums.size();
        if (n == 1) return nums[0];
        if (n == 2) return max(nums[0], nums[1]);
        int foo = nums[0];	// dp[i - 2]
        int bar = max(nums[0], nums[1]); // dp[i - 1]
        int hoge; // dp[i]
        for (int i = 2; i < n; ++i) {
            hoge = max(foo + nums[i], bar);
            foo = bar;
            bar = hoge;
        }
        return hoge;
    }
};

207.课程表

题意

题解

拓扑序裸题

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
class Solution {
public:
    bool canFinish(int numCourses, vector<vector<int>>& prerequisites) {
        vector<vector<int>> G(numCourses);
        vector<int> indegree(numCourses);
        for (auto& i : prerequisites) {
            G[i[1]].push_back(i[0]);
            indegree[i[0]]++;
        }
        int ans = 0;
        stack<int> s;
        for (int i = 0; i < numCourses; ++i) {
            if (indegree[i] == 0) {
                s.push(i);
            }
        }
        while (!s.empty()) {
            auto tmp = s.top();
            s.pop();
            ans++;
            for (int i : G[tmp]) {
                if (--indegree[i] == 0) {
                    s.push(i);
                }
            }
        }
        return ans == numCourses;
    }
};

169.多数元素

题意

题解

记录候选人并计数,计数是候选人比别的数多几个

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
class Solution {
public:
    int majorityElement(vector<int>& nums) {
        int candidate = -1;
        int count = 0;
        for (int i : nums) {
            if (candidate == i) {
                count++;
            } else if (count == 0) {
                candidate = i;
                count = 1;
            } else {
                count--;
            }
        }
        return candidate;
    }
};

208.实现Trie(前缀树)

题意

题解

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
class Trie {
public:
    
    Trie() {
        memset(nx, 0, sizeof(nx));
        memset(exist, 0, sizeof(exist));
    }
    
    void insert(string word) {
        int p = 0;	// 位置
        for (char i : word) {
            int num = i - 'a';
            cout << p << ' ';
            if (!nx[p][num]) nx[p][num] = ++cnt;
            p = nx[p][num];
        }
        exist[p] = true;
    }
    
    bool search(string word) {
        int p = 0;
        for (char i : word) {
            int num = i - 'a';
            if (!nx[p][num]) return false;
            p = nx[p][num];
        }
        return exist[p];
    }
    
    bool startsWith(string prefix) {
        int p = 0;
        for (char i : prefix) {
            int num = i - 'a';
            if (!nx[p][num]) return false;
            p = nx[p][num];
        }
        return true;
    }
private:
    int nx[2005 * 26][26];	// 表示第i个节点的第j个分支是哪个节点,不存在就创建
    int cnt = 0;	// 当前创建了几个节点
    bool exist[2005 * 26];	// 表示第i个节点是否是单词末尾
};

/**
 * Your Trie object will be instantiated and called as such:
 * Trie* obj = new Trie();
 * obj->insert(word);
 * bool param_2 = obj->search(word);
 * bool param_3 = obj->startsWith(prefix);
 */

347.前K个高频元素

题意

题解

堆,用哈希表求次数,按次数排序

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
class Solution {
public:
    vector<int> topKFrequent(vector<int>& nums, int k) {
        unordered_map<int, int> mp;
        for (int i : nums) {
            mp[i]++;
        }
        priority_queue<pair<int, int>, vector<pair<int, int>>, greater<>> pq;	// 小顶堆
        for (auto& [x, y] : mp) {
            if ((int) pq.size() < k) {
                pq.emplace(y, x);
            } else {
                if (y > pq.top().first) {
                    pq.pop();
                    pq.emplace(y, x);
                }
            }
        }
        vector<int> ans;
        while (!pq.empty()) {
            ans.emplace_back(pq.top().second);
            pq.pop();
        }
        return ans;
    }
};

295.数据流的中位数

题意

题解

技巧题

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
class MedianFinder {
public:
    /** initialize your data structure here. */


    priority_queue<int> big;	// 存小数
    priority_queue<int, vector<int>, greater<>> small; // 存大数

    MedianFinder() {

    }
    
    void addNum(int num) {
      	// 先加到堆里
        if (big.empty()) big.push(num);
        //else if (small.empty()) small.push(num);
        else if (num > big.top()) small.push(num);
        else big.push(num);
      
      	// 再调整
        if ((int) big.size() > (int) small.size() + 1) {
            small.push(big.top());
            big.pop();
        }
        if ((int) small.size() > (int) big.size() + 1) {
            big.push(small.top());
            small.pop();
        }
    }
    
    double findMedian() {
        if (big.empty() and small.empty()) return 0;
        if ((int) big.size() > (int) small.size()) return big.top();
        if ((int) small.size() > (int) big.size()) return small.top();
        return (big.top() + small.top()) / 2.0;
    }
};

/**
 * Your MedianFinder object will be instantiated and called as such:
 * MedianFinder* obj = new MedianFinder();
 * obj->addNum(num);
 * double param_2 = obj->findMedian();
 */

5.最长回文子串

题意

题解

Manacher可以达到O(n)

dp

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
class Solution {
public:
    bool dp[1005][1005];
    string longestPalindrome(string s) {
        int n = s.length();
        for (int i = 1; i <= n; ++i) {	// len
            for (int j = 0; j <= n - i; ++j) {
                if (i == 1) {
                    dp[j][j] = true;
                } else if (i == 2) {
                    if (s[j] == s[j + 1]) {
                        dp[j][j + 1] = true;
                    }
                } else {
                    if (s[j] == s[j + i - 1]) {
                        dp[j][j + i - 1] = dp[j + 1][j + i - 2];
                    }
                }
            }
        }
        int ans = 0;
        int l, r;
        for (int i = 0; i < n; ++i) {
            for (int j = i; j < n; ++j) {
                if (dp[i][j]) {
                    if (j - i > r - l) {
                        l = i, r = j;
                        ans = r - l + 1;
                    }
                }
            }
        }
        return s.substr(l, r - l + 1);
    }
};

213.打家劫舍II

题意

题解

与1不同的是,它是环形数组

分两种情况,第0个数必选,第0个数必不选

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
func rob(nums []int) int {
	n := len(nums)
	if n == 1 {
		return nums[0]
	}
	if n == 2 {
		return max(nums[0], nums[1])
	}
	if n == 3 {
		return max(max(nums[0], nums[1]), nums[2])
	}
	dp := make([]int, n + 3)
	ans := 0
	dp[0] = nums[0]
	dp[1] = dp[0]
	for i := 2; i < n - 1; i++ {
		dp[i] = max(dp[i - 1], dp[i - 2] + nums[i])
	}
	ans = max(ans, dp[n - 2])
	for i := range dp {
		dp[i] = 0
	}
	dp[0] = 0
	dp[1] = nums[1]
	for i := 2; i < n; i++ {
		dp[i] = max(dp[i - 1], dp[i - 2] + nums[i])
	}
	ans = max(ans, dp[n - 1])
	return ans
}

337.打家劫舍III

题意

题解

dfs,对于当前节点有选和不选两种情况,分别返回选这个节点的答案和不选节点的答案

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
/**
 * Definition for a binary tree node.
 * type TreeNode struct {
 *     Val int
 *     Left *TreeNode
 *     Right *TreeNode
 * }
 */
func rob(root *TreeNode) int {
	var dfs func(node *TreeNode) (int, int)
	dfs = func(node *TreeNode) (int, int) {
		if node == nil {return 0, 0}
		l1, l0 := dfs(node.Left)
		r1, r0 := dfs(node.Right)
		one := node.Val
		one += l0 + r0
		zero := max(l1, l0) + max(r1, r0)
		return one, zero
	}
	one, zero := dfs(root)
	return max(one, zero)
}

2560.打家劫舍IV

题意

题解

二分

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
class Solution:
    def minCapability(self, nums: List[int], k: int) -> int:
        n = len(nums)
        l, r = 1, 10**9 + 3
        def ok(m: int) -> bool:
            tmp = []
            for i in range(n):
                if nums[i] <= m:
                    tmp.append(i)
            cnt = 1
            if len(tmp) == 0:
                return False
            now = tmp[0]
            for i in tmp:
                if i == tmp[0]:
                    continue
                if i != now + 1:
                    cnt += 1
                    now = i
            return cnt >= k
        while l <= r:
            mid = (l + r) // 2
            if ok(mid):
                r = mid - 1
            else:
                l = mid + 1
        return l

43.字符串相乘

题意

题解

看官方题解方法二,时间复杂度O(nm)

还可以用fft

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
class Solution {
public:
    string multiply(string num1, string num2) {
        if (num1 == "0" or num2 == "0") return "0";
        int n = (int) num1.size(), m = (int) num2.size();
        vector<int> ans(m + n, 0);
        for (int i = n - 1; i >= 0; --i) {
            int x = num1[i] - '0';
            for (int j = m - 1; j >= 0; --j) {
                int y = num2[j] - '0';
                ans[i + j + 1] += x * y;
            }
        }
        for (int i = n + m - 1; i > 0; --i) {
            ans[i - 1] += ans[i] / 10;
            ans[i] %= 10;
        }
        // !!
        int index = ans[0] == 0 ? 1 : 0;
        string fans;
        while (index < m + n) {
            fans += ans[index] + '0';
            index++;
        }
        return fans;
    }
};