LeetCode算法——双指针篇

宫侑的发球最终进化为三刀流,那么我的题解也未必要循规蹈矩!


1、验证回文串

题目描述:

解法:

这题官方给的关于双指针的题解都用到了多个库函数,如tolower (大写字母转小写)、isalnum(判断一个字符是否是 字母 或者 十进制数字 )

我想避免使用库函数,所有就不用双指针的做法了,而是使用

思路很简单:

1、先将字符串中所有字母统一为小写字母

2、将数字和小写字母入栈,同时将完整的、连续的字符都加入到一个新字符串newS中

3、再使用一个新字符串stkPop接收栈每一次pop出的字符

4、比较 newS 和 stkPop 是否相等

这道题的描述中提到了"字母和数字都属于字母数字字符",但示例中没有给出串中包含数字的情况,导致测试用例**'0P'** 没有通过,这也是因为我忽略了串中还有数字的情况

要点:

1、对于大写字母,在 ASCII 码的基础上 +32 就能转为小写字母

2、栈从top位置开始pop,每次pop后栈中的元素会被删除

cpp 复制代码
class Solution {
public:
    bool isPalindrome(string s) {
        
        stack <char> stk;
        string newS;
        string stkPop;  // 接收栈中所有字符pop后的结果

        if(s.size() <= 1) return true;
        for(int i = 0; i < s.size(); ++i){
            if(s[i] >= 'A' && s[i] <= 'Z'){
                s[i] += 32;  // 先将串中所有大写字母转为小写
            }

            // 只将串中的字母和数字入栈
            if((s[i] >= 'a' && s[i] <= 'z') || (s[i] >= '0' && s[i] <= '9')){
                stk.push(s[i]); 
                newS += s[i];
            }
        }
        while(!stk.empty()){
            stkPop += stk.top();
            stk.pop();
        }
        return stkPop == newS;
    }
};

2、判断子序列

题目描述:

解法:

看到这种题,第一反应就是2个 for 循环,这种想法很不好

思路:

为串 s 的首字符设置一个指针,同理为串 t 的首字符也设置一个指针

我们是要在 t 中找出和 s 可以匹配上的子串,当两个串中匹配上了同一个字符时,两个指针都向后走一步,以便进行下一个字符的匹配

若 s 的第2个字符与 t 的第2个字符没有匹配上,此时指向 s 的指针不动,指向 t 的指针继续向后移动,直到找到匹配项

当 s 的指针走完整个 s 串时,表示在 t 中有可以匹配上的子序列,而 s 的指针没有走完 s 串时,则没有匹配成功

cpp 复制代码
class Solution {
public:
    bool isSubsequence(string s, string t) {
        
        // 指向字符串s、t首字符的指针
        int i = 0, j = 0;
        if(s.size() == 0) return true;

        for(i, j; j < t.size(); ++j){
            if(s[i] == t[j]){
                i++;  // 匹配上了, i 和 j同走,没匹配上,只有 j 往后走
                if(i == s.size()) return true;
            }
        }
        return false;
    }
};

3、两数之和Ⅱ-输入有序数组

题目描述:

解法:

被题目描述中的"给你一个下标从1开始的整数数组"迷惑了,一开始设定的指针是

int i = 1;

int j = numbers.size();

哈哈哈哈果然溢出了,不论何种描述,数组容器 的下标在计算机世界里永远是从0开始的

最终根据题目要求返回 **{i + 1, j + 1}**就好了

之前只知道python中单个函数可以返回多个值 ,还在想C++怎么返回两个下标值呢?刚开始甚至还定义了另一个 vector<int> res 来接收下标,最终看了题解后发现了 **return {i + 1, j + 1}**这种写法

cpp 复制代码
class Solution {
public:
    vector<int> twoSum(vector<int>& numbers, int target) {

        int i = 0;  // 首指针
        int j = numbers.size() - 1;  // 尾指针

        for(i, j; i < numbers.size(), j >= 0;){
            if((numbers[i] + numbers[j]) > target){
                j--;  // 两数之和大于目标,尾指针向前走
            }
            else if((numbers[i] + numbers[j]) < target){
                i++;
            }
            else{
                if((numbers[i] + numbers[j]) == target) return {i + 1, j + 1};
            }
        }
        return {-1, -1};
    }
};

4、盛水最多的容器

题目描述:

官方写的垂线、端点之类的术语属实费脑子,推荐大家去看K神的题解------(传送门)

解法:

实际上求解的正是++矩形面积++,规定好 i < j 后,唯一要做的就是判断 i 和 j 代表的槽板哪个高哪个低,因为较小的一方才能盛水而不漏

每次移动槽板较低的一方(此处可能用到了贪心的思想),通过不断比较原来面积和新面积的大小来返回最终的最大面积。

cpp 复制代码
class Solution {
public:
    int maxArea(vector<int>& height) {
        
        int res = 0;
        int i = 0;
        int j = height.size() - 1;

        while(i < j){

            // 指针指向的水槽板高度分别为 h[i], h[j],两者中小的一方决定容纳面积
            res = max(res, (min(height[i], height[j]) * (j - i)));
            if(height[i] < height[j]) i++;
            
            else j--;
        }
        return res;
    }
};
相关推荐
JNU freshman39 分钟前
力扣第435场周赛讲解
算法·leetcode·蓝桥杯
眼镜哥(with glasses)40 分钟前
蓝桥杯python基础算法(2-2)——基础算法(B)——模拟(上)
算法
赵鑫亿2 小时前
7.DP算法
算法·dp
iqay2 小时前
【C语言】填空题/程序填空题1
c语言·开发语言·数据结构·c++·算法·c#
还有糕手3 小时前
算法【有依赖的背包】
算法·动态规划
pursuit_csdn3 小时前
力扣 347. 前 K 个高频元素
算法·leetcode
wen__xvn3 小时前
每日一题洛谷B3865 [GESP202309 二级] 小杨的 X 字矩阵c++
c++·算法·矩阵
makabaka_T_T4 小时前
25寒假算法刷题 | Day1 | LeetCode 240. 搜索二维矩阵 II,148. 排序链表
数据结构·c++·算法·leetcode·链表·矩阵
辞半夏丶北笙4 小时前
最近最少使用算法(LRU最近最少使用)缓存替换算法
java·算法·缓存
BingLin-Liu4 小时前
蓝桥杯备考:六大排序算法
算法·排序算法