leetcode 3.1

leetcode hot 100

双指针

1.三数之和

三数之和

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

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. 左侧,只有当前的高度heightl < l_max时才能够有积水,否则例如 0,1,2,3,4,5是没办法形成积水的
  5. 同理,右侧,只有当前的高度heightr < 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:dpm + 1n + 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相同所以 dp11 = 1

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

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

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

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

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

因此可以得出:

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

dpi + 1j + 1 = max(dpi+1j, dpij+1);

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

3)边界条件:dp0X = 0, dpX0 = 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];
    }
};
相关推荐
lqqjuly3 分钟前
前沿算法深度解析(二)
人工智能·算法·机器学习
徐小夕1 小时前
万字长文!千万级文档 RAG 知识库系统落地实践
前端·算法·github
akunkuntaimei2 小时前
2026年高考数学各省真题及答案(完整版)
算法·高考
Hello:CodeWorld2 小时前
C 风格变参 vs C++ 变参模板:核心区别与选型指南
c语言·c++·算法
8Qi84 小时前
LeetCode 516:最长回文子序列
算法·leetcode·职场和发展·动态规划
youngerwang5 小时前
【从搬运工到协处理器:网卡芯片架构、算法、验证与边缘演进深度剖析】
网络·算法·架构·芯片
KaMeidebaby5 小时前
卡梅德生物技术快报|纯化重组蛋白实操详解
人工智能·python·tcp/ip·算法·机器学习
手写码匠6 小时前
从零实现 Prompt 工程引擎:结构化提示、自动优化与多轮自省体系
人工智能·深度学习·算法·aigc
无限码力6 小时前
阿里算法岗 0530笔试真题 - 多约束条件下的元素匹配统计
算法·阿里笔试真题·阿里机试真题·阿里算法岗笔试
lqqjuly6 小时前
MLA — 多头潜在注意力深度解析
深度学习·神经网络·算法