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;
    }
};
相关推荐
core5128 分钟前
PageRank 算法:互联网的“人气投票”
算法·pagerank
小白菜又菜12 分钟前
Leetcode 1523. Count Odd Numbers in an Interval Range
算法·leetcode
你们补药再卷啦39 分钟前
人工智能算法概览
人工智能·算法
cnxy1881 小时前
围棋对弈Python程序开发完整指南:步骤3 - 气(Liberties)的计算算法设计
python·算法·深度优先
AndrewHZ1 小时前
【图像处理基石】什么是光栅化?
图像处理·人工智能·算法·计算机视觉·3d·图形渲染·光栅化
小白菜又菜1 小时前
Leetcode 944. Delete Columns to Make Sorted
算法·leetcode
我找到地球的支点啦2 小时前
Matlab系列(006) 一利用matlab保存txt文件和读取txt文件
开发语言·算法·matlab
Dev7z2 小时前
基于Matlab实现GRACE卫星重力数据的全球水储量变化估算与分析
人工智能·算法·matlab
爱喝热水的呀哈喽2 小时前
11题目汇总
算法
三斗米3 小时前
Transformer入门:一文读懂《Attention Is All You Need》
算法·架构