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;
    }
};
相关推荐
疋瓞31 分钟前
C++_STL和数据结构《1》_STL、STL_迭代器、c++中的模版、STL_vecto、列表初始化、三个算法、链表
数据结构·c++·算法
JJJJ_iii33 分钟前
【左程云算法09】栈的入门题目-最小栈
java·开发语言·数据结构·算法·时间复杂度
Bear on Toilet1 小时前
继承类模板:函数未在模板定义上下文中声明,只能通过实例化上下文中参数相关的查找找到
开发语言·javascript·c++·算法·继承
金融小师妹1 小时前
多因子AI回归揭示通胀-就业背离,黄金价格稳态区间的时序建模
大数据·人工智能·算法
程序员东岸1 小时前
C语言入门指南:字符函数和字符串函数
c语言·笔记·学习·程序人生·算法
小猪咪piggy2 小时前
【算法】day2 双指针+滑动窗口
数据结构·算法·leetcode
max5006002 小时前
OpenSTL PredRNNv2 模型复现与自定义数据集训练
开发语言·人工智能·python·深度学习·算法
budingxiaomoli2 小时前
AVL树知识总结
数据结构·算法
jz-炸芯片的zero2 小时前
【Zephyr电源与功耗专题】14_BMS电池管理算法(三重验证机制实现高精度电量估算)
单片机·物联网·算法·zephyr·bms电源管理算法
u0104058364 小时前
京东返利app的分布式ID生成策略:雪花算法在订单系统中的实践
分布式·算法