ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

双指针技巧秒杀七道数组题目

2022-04-05 10:35:00  阅读:279  来源: 互联网

标签:slow nums int fast 秒杀 数组 七道 指针


双指针技巧秒杀七道数组题目

在处理数组和链表相关问题时,双指针技巧是经常用到的,双指针技巧主要分为两类:左右指针快慢指针

所谓左右指针,就是两个指针相向而行或者相背而行;而所谓快慢指针,就是两个指针同向而行,一快一慢。

一、快慢指针技巧

数组问题中比较常见且难度不高的的快慢指针技巧,是让你原地修改数组

力扣第 26 题「 删除有序数组中的重复项

我的C++代码:

让慢指针 slow 走在后面,快指针 fast 走在前面探路,找到一个不重复的元素就赋值给 slow 并让 slow 前进一步。

这样,就保证了 nums[0..slow] 都是无重复的元素,当 fast 指针遍历完整个数组 nums 后,nums[0..slow] 就是整个数组去重之后的结果。

class Solution {
public:
    int removeDuplicates(vector<int>& nums) {
        int slow=0,fast=1;
        while(fast<nums.size()){
            if(nums[fast]!=nums[slow]){
                slow++;
                nums[slow]=nums[fast];
            }
            fast++;
        }
        return slow+1;
    }
};

算法执行的过程如下 GIF 图:

img

力扣第 83 题「 删除排序链表中的重复元素

与上题思路相同,或许更简单些?

class Solution {
public:
    ListNode* deleteDuplicates(ListNode* head) {
        if(head==nullptr)return nullptr;
        ListNode* slow=head;
        ListNode* fast=head;
        while(fast!=nullptr){
            if(fast->val!=slow->val)
            {
                slow->next=fast;
                slow=fast;
            }
            fast=fast->next;
        }
        slow->next=fast;
        return head;
    }
};

算法执行的过程请看下面这个 GIF:

img

力扣第 27 题「 移除元素

我的C++代码

如果 fast 遇到值为 val 的元素,则直接跳过,否则就赋值给 slow 指针,并让 slow 前进一步。

注意这里和有序数组去重的解法有一个细节差异,我们这里是先给 nums[slow] 赋值然后再给 slow++,这样可以保证 nums[0..slow-1] 是不包含值为 val 的元素的,最后的结果数组长度就是 slow

class Solution {
public:
    int removeElement(vector<int>& nums, int val) {
        int slow=0,fast=0;
        while(fast<nums.size()){
           if(nums[fast]!=val){
               nums[slow]=nums[fast];
               slow++;

           }
            
            fast++;
        }
        return slow;
    }
};

力扣第 283 题「 移动零

我的C++代码:

相当于移除 nums 中的所有 0,然后再把后面的元素都赋值为 0 即可。

class Solution {
public:
    void moveZeroes(vector<int>& nums) {
        int slow=0,fast=0;
        while(fast<nums.size()){
            if(nums[fast]!=0){
                nums[slow]=nums[fast];
                slow++;
            }
            fast++;
        }
        while(slow<nums.size()){
            nums[slow]=0;
            slow++;
        }
    }
};

二、左右指针的常用算法

1、二分查找

框架

int binarySearch(int[] nums, int target) {
    // 一左一右两个指针相向而行
    int left = 0, right = nums.length - 1;
    while(left <= right) {
        int mid = (right + left) / 2;
        if(nums[mid] == target)
            return mid; 
        else if (nums[mid] < target)
            left = mid + 1; 
        else if (nums[mid] > target)
            right = mid - 1;
    }
    return -1;
}

2、两数之和

力扣第 167 题「 两数之和 II

C++代码:

只要数组有序,就应该想到双指针技巧。这道题的解法有点类似二分查找,通过调节 leftright 就可以调整 sum 的大小:

class Solution {
public:
    vector<int> twoSum(vector<int>& numbers, int target) {
        //左右双指针 大了就调整右指针 小了就调整左指针
        int le=0,ri=numbers.size()-1;
        while(le<ri){
            if(numbers[le]+numbers[ri]==target){
                return {le+1,ri+1};
            }
            //和小了
            if(numbers[le]+numbers[ri]<target){
                le++;
            }else{//和大了
                ri--;
            }
        }
        return {le+1,ri+1};
    }
};

3、反转数组

一般编程语言都会提供 reverse 函数,其实这个函数的原理非常简单,力扣第 344 题「 反转字符串」就是类似的需求,让你反转一个 char[] 类型的字符数组

void reverseString(char[] s) {
    // 一左一右两个指针相向而行
    int left = 0, right = s.length - 1;
    while (left < right) {
        // 交换 s[left] 和 s[right]
        char temp = s[left];
        s[left] = s[right];
        s[right] = temp;
        left++;
        right--;
    }
}

4、回文串判断

首先明确一下,回文串就是正着读和反着读都一样的字符串。

比如说字符串 abaabba 都是回文串,因为它们对称,反过来还是和本身一样;反之,字符串 abac 就不是回文串。也是两个指针相向而行即可

力扣第 5 题「 最长回文子串」:

C++代码:

最长回文子串使用的左右指针和之前题目的左右指针有一些不同:之前的左右指针都是从两端向中间相向而行,而回文子串问题则是让左右指针从中心向两端扩展。

class Solution {
public:
    string longestPalindrome(string s) {
        int sz=s.size();
        string res="";
        for(int i=0;i<sz;i++){
            string a1=Palindrome(s,i,i);
            string a2=Palindrome(s,i,i+1);
            res=res.size()>a1.size()?res:a1;
            res=res.size()>a2.size()?res:a2;
        }
        return res;
    }
    string Palindrome(string s,int le,int ri){
        // 防止索引越界
        while (le >= 0 && ri < s.size()&& s[le] == s[ri]) {
        // 双指针,向两边展开
             le--; ri++;
        }
        // 返回以 s[l] 和 s[r] 为中心的最长回文串
        return s.substr(le+1,ri-le-1);
    }
};

标签:slow,nums,int,fast,秒杀,数组,七道,指针
来源: https://www.cnblogs.com/BailanZ/p/16101584.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有