codetop标签双指针题目大全解析(三),双指针刷穿地心!!!!!

news2024/11/24 0:28:53

复习比学习更重要,更需要投入时间,更需要花费精力

  • 1.字符串的排列
  • 2.找出字符串中第一个匹配的下标
  • 3.最大连续1的个数II
  • 4.数组中的山脉
  • 5.移除元素
  • 6.两个数组的交集II
  • 7.有序数组的平方
  • 8.删除有序数组中的重复项II
  • 9.寻找重复数
  • 10.水果成篮

1.字符串的排列

在滑动窗口总结文章里面讲解过了

二刷debug:首先,缩小窗口的条件是r-l >s1.size()。然后,必须用need.count( c )而不是need[c] >= 1。count( c ) 仅仅检查键 c 是否存在于 unordered_map 中,与键对应的值无关!而need[c]>=1是必须need中有其键并且数量大于等于1

class Solution {
public:
    bool checkInclusion(string s1, string s2) {
        unordered_map<char, int> window, need;
        for(char c : s1) need[c] ++;
        int left = 0, right = 0;
        int valid = 0;
        while(right < s2.size()){
            char c = s2[right];
            right++;
            if(need.count(c)){
                window[c]++;
                if(need[c] == window[c]) valid++;
            }
            while(right - left > s1.size()){
                char d = s2[left];
                left ++;
                if(need.count(d)){
                    if(need[d] == window[d]) valid--;
                    window[d]--;
                }
            }
            if(need.size() == valid && right - left == s1.size()){
                return true;
            }
        }
        return false;
    }
};

2.找出字符串中第一个匹配的下标

在这里插入图片描述
拿到手的第一想法就是,滑动窗口,输出left
但是这个要求顺序完全一样,不能是排列或者组合
查了一下KMP是专门弄这种的,学习新算法了在这里插入图片描述(我只是来做双指针的…)这篇文章是个纯刷题记录,不贴详细讲解,最多记录大致思路,需要讲解去秒杀直接部分->传送门
二刷debug:写出来了,几乎是靠背的,注意ne初始化

class Solution {
public:
    int strStr(string haystack, string needle) {
        int n = haystack.size();
        int m = needle.size();
        vector<int> ne(m, -1);// ne数组必须初始化为-1而不是0,只有-1才代表没有相同的前后缀
        // 建next数组
        for(int i = 1, j = -1; i < m; i ++){
            while(j != -1 && needle[i] != needle[j + 1]) j = ne[j];
            if(needle[i] == needle[j + 1]) j ++;
            ne[i] = j;
        }
        // 匹配
        for(int i = 0, j = -1; i < n; i ++){
            while(j != -1 && haystack[i] != needle[j + 1]) j = ne[j];
            if(haystack[i] == needle[j + 1]) j ++;
            if(j == m - 1){
                return i - m + 1;
            }
        }
        return -1;

    }
};

3.最大连续1的个数II

不是,家人们,滑动窗口为什么都划到双指针标签下了啊
题:
给定一个二进制数组 nums 和一个整数 k,如果可以翻转最多 k 个 0 ,则返回 数组中连续 1 的最大个数 。
eg:
输入:nums = [1,1,1,0,0,0,1,1,1,1,0], K = 2
输出:6
在秒杀系列的滑动窗口秒杀文章里面写过
用滑动窗口做题需要先明白3个问题

  1. 什么时候扩大窗口?更改什么数据?
  2. 什么时候缩小窗口?更改什么数据?
  3. 什么时候得到答案?

针对123的答案:

  1. 当可替换次数k>=0的时候扩大窗口,更改窗口里面1的个数,让窗口里面都是1,等于0的时候也扩,万一窗口外面不需要改呢。
  2. 当可替换次数k<0的时候缩小窗口,可替换次数++,以便继续扩大
  3. k>=0的时候,窗口内部都是1,len更新
class Solution {
public:
    int longestOnes(vector<int>& nums, int k) {
        int left = 0, right = 0;
        int windowOneCount = 0;
        int res = 0;
        while(right < nums.size()){
            //right是0也++,是1就windowOneCount++,自身也++
            if(nums[right] == 1){
                windowOneCount ++;
            }
            right ++;
            //窗口里面0的个数超过了k,就开始缩小窗口
            while(right - left - windowOneCount > k){
                if(nums[left] == 1) windowOneCount --;
                left ++;
            }

            res = max(res, right - left);
        }
        return res;
    }
};

4.数组中的山脉

在这里插入图片描述
先找到可能得山顶,再双指针两边扩展,记录res
留意l,r,i的边界
二刷debug:注意双指针两边扩展的手法,还有边界问题,如果是只能到倒数第二个元素的话,i<.size()-1即可

class Solution {
public:
    int longestMountain(vector<int>& arr) {
        int l = 0, r = 0, res = 0;
        for(int i = 1; i < arr.size() - 1; i ++){
            if(arr[i] > arr[i - 1] && arr[i] > arr[i + 1]){
                l = i - 1;
                r = i + 1;
                while(l > 0 && arr[l] > arr[l - 1]) l --;
                while(r < arr.size() - 1 && arr[r] > arr[r + 1]) r ++;
                res = max(res, r - l + 1);
            }
        }
        return res;
    }
};

5.移除元素

移除val,返回新数组的长度
双指针里也有这题,秒啦

class Solution {
    public int removeElement(int[] nums, int val) {
        int i = 0;
        for (int n : nums)
            if (n != val) {
                nums[i] = n;
                i++;
            }
        return i;
    }
}

6.两个数组的交集II

给nums1和nums2,以数组的形式返回两数组里都存在的数,并且这个数的次数要等于两个数组中这个数出现次数更少的那个

别人的代码是真的优雅
这个代码先记录了nums1中每个元素出现的次数到umap中
再在nums2中for每个元素
如果元素在umap中有记录则将其push进res,并且umap记录数–

假如nums1中才是数字出现少的那个,那么umap[nums1]会先到0,以至于res不了nums2的元素,假如nums2才是数字出现少的那个,那么if(nums2)会先空

太优雅了o(╥﹏╥)o

class Solution {
public:
    vector<int> intersect(vector<int>& nums1, vector<int>& nums2) {
        unordered_map<int, int> umap;
        vector<int> res;
        for(int i = 0; i < nums1.size(); i ++) umap[nums1[i]]++;
        for(int i = 0; i < nums2.size(); i ++){
            if(umap[nums2[i]]){
                res.push_back(nums2[i]);
                umap[nums2[i]] --;
            }
        }
        return res;
    }
};

7.有序数组的平方

给一个非递减的数组,现在需要你将每个元素都平方,然后递增排序,返回nums。注意,需要时间复杂度是O(n)

sort的家伙,以后面试也排人后面!!!!

sort的复杂度是O(nlogn),所以不能用sort,只能用双指针

这里有个十分关键的点,就是:原本的数组本身就是有序的,是一个非递减的数组,那么即使数组中元素有正有负,绝对值最大的元素肯定是在数组的两端的,即数组平方的最大值是在数组的两端的。

所以我们可以使用两个双指针i,j一个指向起始位置,一个指向数组的末尾。
定义一个新的数组result用于储存新的有序平方后的元素。

class Solution {
public:
 //双指针
    vector<int> sortedSquares(vector<int>& A) {
        int k = A.size() - 1; //指向新数组的末尾,从后往前赋值
        vector<int> result(A.size(), 0);
        for (int i = 0, j = A.size() - 1; i <= j;) { // 注意这里要i <= j,因为最后要处理两个元素
            if (A[i] * A[i] < A[j] * A[j]) { //判断条件1:尾部元素更大
                result[k--] = A[j] * A[j];
                j--;
            }
            else {
                result[k--] = A[i] * A[i]; //判断条件2:头部元素更大
                i++;
            }
        }
        return result;
    }
};

8.删除有序数组中的重复项II

给你一个有序数组 nums ,请你 原地 删除重复出现的元素,使得出现次数超过两次的元素只出现两次 ,返回删除后数组的新长度。

不要使用额外的数组空间,你必须在 原地 修改输入数组 并在使用 O(1) 额外空间的条件下完成。

前2个肯定不用删,所以可以跳过,从j = 2开始比
还是太优雅了这代码

二刷debug:不会,很难理解

class Solution {
public:
    int removeDuplicates(vector<int>& nums) {
        if(nums.size() <= 2) return nums.size();
        int i = 2;
        for(int j = 2; j < nums.size(); j ++){
            if(nums[j] != nums[i - 2]){
                nums[i] = nums[j];
                i ++;
            }
        }
        return i;
    }
};

9.寻找重复数

在这里插入图片描述

不可以用sort也不可以用额外数组
这个要求真的是把我路都堵死了…
二刷debug:不会…
数组小技巧:数组也可以看做链表来做
以图为例,天然就有数组链表0->1->3->2->4
fast = nums[nums[fast]]相当于fast = fast->next->next
slow = nums[slow]相当于slow = slow->next
在这里插入图片描述
几刷?:容易写成return nums[slow],实际上最后一个是slow = nums[slow],所以直接写成return slow就可以了

class Solution {
public:
    int findDuplicate(vector<int>& nums) {
        int fast = 0, slow = 0;
        while(true){
            fast = nums[nums[fast]];
            slow = nums[slow];
            if(fast == slow) break;
        }
        slow = 0;
        while(fast != slow){
            fast = nums[fast];
            slow = nums[slow];
        }
        return slow;
    }
};

10.水果成篮

fruits数组,fruits[i]代表一种水果,比如fruits[2] = 1,,代表香蕉
现在有fruits.size()棵水果树,每次只能摘一颗树
现在有2个篮子,每个篮子装一种水果,问最多能摘多少棵数

fruit = [1,2,1],有两种水果树,所以能摘三棵,都能摘,篮子装得下

滑动窗口。要注意不需要window.size() == need,也要计算len,因为有while(window.size() > need)在,窗口不是小了就是刚刚好,不可能大,如果fruit的水果树种类本来就不足2个,就可以返回
另外,当缩小窗口,导致其中一个苹果树没了,window应该erase掉。否则还占用一个size

二刷debug:小于等于2的水果树种类也可以,另外unordered_map类型可以使用erase

class Solution {
public:
    int totalFruit(vector<int>& fruits) {
        unordered_map<int, int> window;
        int need = 2;
        int len = 0;
        int left = 0, right = 0;
        while(right < fruits.size()){
            int c = fruits[right];
            right++;
            window[c]++;;
            while(window.size() > need){
                int d = fruits[left];
                left++;
                window[d]--;
                if(window[d] == 0) window.erase(d);
            }
            len = max(len, right - left);
            
        }
        return len;
    }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2196695.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

HUAWEI_HCIA_实验指南_Lib1.4_配置通过Telnet登录系统

一、原理概述 Telnet(Telecommunication Network Protocol)起源于ARPANET,是最早的Internet应用之一。 Telnet 通常用在远程登录应用中&#xff0c;以便对本地或远端运行的网络设备进行配置、监控和维护。如网络中有多台设备需要配置和管理&#xff0c;用户无需为每一台设备…

C++ 算法学习——7.4.1 优化算法——双指针

双指针法&#xff08;Two Pointers&#xff09;是一种常用的算法技巧&#xff0c;通常用于解决数组或链表中的问题。这种技巧通过维护两个指针&#xff0c;通常分别指向数组或链表的不同位置&#xff0c;来协同解决问题。双指针法一般有两种类型&#xff1a;快慢指针和左右指针…

查询计算移出数据库用 Java 太慢咋办

很多现代应用会把数据计算和处理任务从数据库移出来采用 Java 实现&#xff0c;这样能获得架构上的好处&#xff0c;而且 Java 有完善过程处理能力&#xff0c;应对日益复杂的业务逻辑比 SQL 更得心应手&#xff08;虽然代码不短&#xff09;。不过&#xff0c;我们常常会发现&…

为什么没有能够处理 Python 字节码的 CPU?

问题 有没有能够处理Python字节码&#xff08;Python bytecode&#xff09;的CPU&#xff1f;众所周知&#xff0c;CPU靠执行字节码指令运作。那有没有能够处理Python字节码&#xff08;Python bytecode&#xff09;的CPU&#xff1f;如果没有&#xff0c;为什么&#xff1f;不…

OpenHarmony(鸿蒙南向开发)——轻量系统内核(LiteOS-M)【内存管理】

往期知识点记录&#xff1a; 鸿蒙&#xff08;HarmonyOS&#xff09;应用层开发&#xff08;北向&#xff09;知识点汇总 鸿蒙&#xff08;OpenHarmony&#xff09;南向开发保姆级知识点汇总~ 持续更新中…… 基本概念 内存管理模块管理系统的内存资源&#xff0c;它是操作系…

LeetCode 刷题基础 -- 模板原型Ⅰ

模板原型 - 基础篇 学习网站一、进制转换二、二分查找① 查找指定元素② 查找第一个大于等于 x 值的序列下标③ 查找第一个大于 x 值的序列下标④ 单峰序列 三、双指针① 两数之和② 序列合并③ 集合求交④ 集合求并 四、其他高效技巧与算法① 区间和② 01 对③ 左小数 五、数学…

【每日刷题】Day134

【每日刷题】Day134 &#x1f955;个人主页&#xff1a;开敲&#x1f349; &#x1f525;所属专栏&#xff1a;每日刷题&#x1f34d; &#x1f33c;文章目录&#x1f33c; 1. 1218. 最长定差子序列 - 力扣&#xff08;LeetCode&#xff09; 2. LCR 116. 省份数量 - 力扣&…

掌握这17个Python自动化操作,简化你的日常工作流程,提升工作效率!

Python是一种流行的编程语言&#xff0c;以其简单性和可读性而闻名。因其能够提供大量的库和模块&#xff0c;它成为了自动化各种任务的绝佳选择。让我们进入自动化的世界&#xff0c;探索17个可以简化工作并节省时间精力的Python脚本。 目录&#xff08;上篇&#xff09; 1.自…

小型数控铣床助力职业教育教学模式

小型数控铣床是在普通铣床上集成了数字控制系统&#xff0c;可以在程序代码的控制下较精确地进行铣削加工的机床。与普通小型铣床相比&#xff0c;小型数控铣床通过数字控制系统实现了自动化加工&#xff0c;提高了加工精度和效率。 小型数控铣床的引入推动了教育装备的现代化进…

基于单片机的书库环境监测

目录 一、主要功能 二、硬件资源 三、程序编程 四、实现现象 一、主要功能 基于51单片机&#xff0c;采用DHT11湿度传感器检测湿度&#xff0c;DS18B20温度传感器检测温度&#xff0c; 采用滑动变阻器连接数模转换器模拟二氧化碳和氧气浓度检测&#xff0c;各项数值通过lc…

Trickle流量限速工具使用示例

简介&#xff1a;trickle 是一个轻量级的流量限速工具&#xff0c;允许用户限制应用程序的网络带宽使用&#xff0c;以便更好地管理网络资源和优化网络性能。本文将介绍 trickle 的安装和使用方法&#xff0c;并通过 Python 封装示例展示如何使用 trickle 控制网络带宽。 历史…

关于PPT生成的开源大模型总结

目前需要开源的PPT生成模型&#xff0c;在这里对github上的一些模型进行筛选 搜索关键词&#xff1a;ppt generate&#xff08;more starts&#xff09; williamfzc/chat-gpt-ppt: 支持直接生成PPT支持中英文需要调用ChatGPT&#xff08;Add your token (official openai api k…

使用正则表达式删除文本的奇数行或者偶数行

用智谱清言和kimi搜出来的结果都没法在notepad生效&#xff0c;后面在overflow上找到的答案比较靠谱。 查找&#xff1a;^[^\n]*\n([^\n]*) 替换&#xff1a;\1 删除偶数行 查找&#xff1a;^([^\n]*)\n[^\n]* 替换&#xff1a;\1 代码解释 ^&#xff1a;这个符号代表字符…

Excel日期导入数据库变为数字怎么办

在Excel导入到数据库的过程中&#xff0c;经常会碰到Excel里面的日期数据&#xff0c;导进去过后变成了数字。 如下图&#xff1a; 使用navicate等数据库编辑器导入数据库后&#xff1a; 原因分析&#xff1a;这是因为日期和时间在excel中都是以数字形式存储的&#xff0c;这个…

重学SpringBoot3-集成Redis(二)之注解驱动

更多SpringBoot3内容请关注我的专栏&#xff1a;《SpringBoot3》 期待您的点赞&#x1f44d;收藏⭐评论✍ 重学SpringBoot3-集成Redis&#xff08;二&#xff09;之注解驱动 1. 为什么选择 Redis 作为缓存&#xff1f;2. 如何在 Spring Boot 中启用 Redis 缓存&#xff1f;2.1 …

多模态大模型调研BLIP、BLIP2、InstructBLIP

ITC:图像向量与文本向量对齐在同一特征空间 ITM:二分类任务。负样本构建:前方ITC分错的地方&#xff0c;在对比学习的基础上&#xff0c;更细粒度的对其特征。 LM:GPT的生成任务&#xff0c;将文本重新进行预测。 BLIP另一个贡献在于弱监督图文数据清洗方面&#xff0c;通过训…

学习threejs,模拟窗户光源

&#x1f468;‍⚕️ 主页&#xff1a; gis分享者 &#x1f468;‍⚕️ 感谢各位大佬 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍⚕️ 收录于专栏&#xff1a;threejs gis工程师 文章目录 一、&#x1f340;前言二、&#x1f340;绘制任意字体模型…

棒球运动物体检测系统源码分享

棒球运动物体检测系统源码分享 [一条龙教学YOLOV8标注好的数据集一键训练_70全套改进创新点发刊_Web前端展示] 1.研究背景与意义 项目参考AAAI Association for the Advancement of Artificial Intelligence 项目来源AACV Association for the Advancement of Computer Vis…

应用界面编写(十四)

一. 介绍QT 接下来我们会在Qt Creater来进行界面的编写&#xff0c;并且在荔枝派中运行。那么我们有必要了解一下Qt到底是什么呢&#xff1f;它又为什么可以在荔枝派中运行呢&#xff1f; QT是一个跨平台的应用程序和用户界面框架&#xff0c;用于开发具有图形界面的软件。而…