leetcode 3.1

leetcode hot 100

双指针

1.三数之和

三数之和

排序 + 双指针的方法,固定一个数nums[i], 用两数和找target -= nums[i] 的数需要注意两点:

1.需要去掉重复数字

cpp 复制代码
while (l < r && nums[l] == nums[--l]);
while (l < r && nums[r] == nums[++r]);
....
while (i < n - 1 && nums[i] == nums[i + 1]) i++;

2.如果用这种方法去掉重复数字,那么一定要先执行 l++ && r--再去执行去重,防止有不重复数字死循环的情况发生

cpp 复制代码
l++, r--;
while (l < r && nums[l] == nums[l - 1]) l++;
while (l < r && nums[r] == nums[r + 1]) r--;
cpp 复制代码
class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        sort(nums.begin(), nums.end());
        int n = nums.size();
        vector<vector<int>> ans;
        for (int i = 0; i < n; i++) {
            int l = i + 1, r = n - 1;
            int tmptar = -nums[i];
            while (l < r) {
                if (nums[l] + nums[r] < tmptar) {
                    l++;
                } else if (nums[l] + nums[r] > tmptar) {
                    r--;
                } else {
                    ans.push_back({nums[i], nums[l], nums[r]});
                    l++, r--;
                    while (l < r && nums[l] == nums[l - 1]) l++;
                    while (l < r && nums[r] == nums[r + 1]) r--;
                }
            }
            while (i < n - 1 && nums[i] == nums[i + 1]) i++;
        }
        return ans;
    }
};

2.接雨水

接雨水

  1. 图1位置的水位由2,3位置决定
  2. 双指针相向寻找,需要找到当前遍历过的、左右两侧的最大高度l_max, r_max
  3. 如果 l_max 小于 r_max,那么左侧一定能够积水,如果 l_max 大于 r_max,右侧一定能够积水
  4. 左侧,只有当前的高度height[l] < l_max时才能够有积水,否则例如 [0,1,2,3,4,5]是没办法形成积水的
  5. 同理,右侧,只有当前的高度height[r] < r_max时才能够有积水
cpp 复制代码
class Solution {
public:
    int trap(vector<int>& height) {
        int n = height.size();
        int l = 0, r = n - 1, l_max = 0, r_max = 0, ans = 0;
        while (l <= r) {
            if (l_max < r_max) {
                l_max = max(l_max, height[l]);
                if (l_max > height[l]) {
                    ans += l_max - height[l];
                }
                l++; 
            } else {
                r_max = max(r_max, height[r]);
                if (r_max > height[r]) {
                    ans += r_max - height[r];
                }
                r--;
            }
        }
        return ans;
    }
};

多维动态规划

1.最长公共子序列

1)确定状态:对于两个字符串的动态规划问题,套路是通用的。

比如说对于字符串 s1 和 s2,它们的长度分别是 m、n,一般来说都要构造一个这样的 DP table:dp[m + 1][n + 1]

2)转移方程:对于 text1:abcde 和 text2:ace 两个字符串,我们定义两个指针进行遍历 i 和 j。

遍历 text1 长度为 m,定义指针 i,从 0~m。固定 i 指针(i == 1)位置,接下来开始遍历 text2 长度为 n,定义指针 j,从 0~n。

第一次遍历 i = 1, j = 1,两个a相同所以 dp[1][1] = 1

第二次遍历 i = 1, j = 2,a与c不等,也不能是0,这里需转换成 a 与 ac 最长子序列,这里需要把之前的关系传递过来,所以dp[1][2] = 1

第三次遍历 i = 1, j = 3,a与e不相同,把之前的关系传递过来,所以dp[1][3] = 1

text2:ace 已经走完来第一轮,接下来text1:abcde 走到来b字符。

第四次遍历 i = 2, j = 1,就是需要比较ab与a的最长子串,把之前的关系传递过来,所以dp[2][1] = 1

我们会发现遍历两个串字符,当不同时 需要考虑两层遍历前面的值(关系传递),也就是左边和上边的其中较大的值,当相同时,需要考虑各自不包含当前字符串的子序列长度,再加上1。

因此可以得出:

现在对比的这两个字符不相同的,那么我们要取它的「要么是text1往前退一格,要么是text2往前退一格,两个的最大值」

dp[i + 1][j + 1] = max(dp[i+1][j], dp[i][j+1]);

对比的两个字符相同,去找它们前面各退一格的值加1即可:dp[i+1][j+1] = dp[i][j] + 1;

3)边界条件:dp[0][X] = 0, dp[X][0] = 0;

4)计算顺序:先行后列

cpp 复制代码
class Solution {
public:
    int longestCommonSubsequence(string text1, string text2) {
        int n = text1.size(), m = text2.size();
        vector<vector<int>> dp(n + 1, vector<int>(m + 1, 0));
        for (int i = 1; i < n + 1; i++) {
            for (int j = 1; j < m + 1; j++)
                if (text1[i - 1] == text2[j - 1])
                    dp[i][j] = dp[i - 1][j - 1] + 1;
                else 
                    dp[i][j] = max(dp[i - 1][j], dp[i][j -1]);
        }
        return dp[n][m];
    }
};
相关推荐
袖清暮雨4 分钟前
5_SparkGraphX讲解
大数据·算法·spark
Tisfy12 分钟前
LeetCode 3218.切蛋糕的最小总开销 I:记忆化搜索(深度优先搜索DFS)
算法·leetcode·深度优先·题解·记忆化搜索
张明奇-琦玉28 分钟前
Boost之log日志使用
linux·服务器·算法
Kai HVZ1 小时前
《机器学习》——利用OpenCV库中的KNN算法进行图像识别
opencv·算法·机器学习
想要AC的sjh1 小时前
【Leetcode】3159. 查询数组中元素的出现位置
数据结构·算法·leetcode
虽千万人 吾往矣1 小时前
golang LeetCode 热题 100(技巧)-更新中
算法·leetcode·职场和发展
南宫生1 小时前
力扣-数据结构-4【算法学习day.75】
java·数据结构·学习·算法·leetcode
fks1432 小时前
leetcode 354. 俄罗斯套娃信封问题
leetcode
chenziang12 小时前
leetcode hot 100搜索回溯
算法·leetcode·职场和发展
TANGLONG2222 小时前
【初阶数据结构与算法】八大排序算法之归并排序与非比较排序(计数排序)
java·数据结构·c++·算法·面试·蓝桥杯·排序算法