3. 无重复字符的最长子串/438. 找到字符串中所有字母异位词/560. 和为 K 的子数组

news2024/11/25 11:22:48

3. 无重复字符的最长子串

给定一个字符串 s ,请你找出其中不含有重复字符的 最长子串 的长度。

示例 1:

输入: s = "abcabcbb"
输出: 3 
解释: 因为无重复字符的最长子串是 "abc",所以其长度为 3。

思路:想象一下我们自己是怎么判断无重复的,先以第K个字符开始,依次加入字符,直到遇到重复的,这里抽象成代码语言就是:需要两个指针(左右指针去移动),还需要一个能存储字符的数据结构,并且可以判断重复。所以我们就可以采用滑动窗口去处理,对于判断重复用的数据结构为哈希集合(即 C++ 中的 std::unordered_set

        在左指针向右移动的时候,我们从哈希集合中移除一个字符,在右指针向右移动的时候,我们往哈希集合中添加一个字符。

由上面想法可以写出下面代码(个人走的弯路,不想看可以跳过)

class Solution {
public:
    int lengthOfLongestSubstring(string s) {
       unordered_set<char> str;
       int right=0;//右值针
       int lenght=0;//最大长度
       for(int i=0;i<s.size();i++){
            str.insert(s[i]);
            right=i+1;
            while(!str.count(s[right])&&right<s.size()){
                str.insert(s[right]);
                ++right;
            }
            lenght=max(lenght,right-i);
            right=i;
            str.clear();
       }
       return lenght;
    }
};

可以解出来,但是耗时太长没有利用好滑动窗口减少比对次数

        我们继续优化,减少比对次数,实际上每次左值针更新时,右指针可以不用左值针处开始比较,可以先将左值针移动向里收缩,右指针不动。

        原因:假设我们选择字符串中的第 k 个字符作为起始位置,并且得到了不包含重复字符的最长子串的结束位置为 rk​。那么当我们选择第 k+1个字符作为起始位置时,首先从 k+1到 rk的字符显然是不重复的,并且由于少了原本的第 k 个字符,我们可以尝试继续增大 rk (而不是从左值针处开始),直到右侧出现了重复字符为止。

优化代码:

class Solution {
public:
    int lengthOfLongestSubstring(string s) {
       unordered_set<char> str;
       int right=0;//右值针
       int lenght=0;//最大长度
       for(int i=0;i<s.size();i++){
            if(i!=0){
                str.erase(s[i-1]);
            }
            while(!str.count(s[right])&&right<s.size()){
                str.insert(s[right]);
                ++right;
            }
            lenght=max(lenght,right-i);
       }
       return lenght;
    }
};

438. 找到字符串中所有字母异位词

给定两个字符串 s 和 p,找到 s 中所有 p 的 异位词 的子串,返回这些子串的起始索引。不考虑答案输出的顺序。

异位词 指由相同字母重排列形成的字符串(包括相同的字符串)。

示例 1:

输入: s = "cbaebabacd", p = "abc"
输出: [0,6]
解释:
起始索引等于 0 的子串是 "cba", 它是 "abc" 的异位词。
起始索引等于 6 的子串是 "bac", 它是 "abc" 的异位词。

思路:用滑动窗口在S中找到和p的异位词就行,滑动过程中维持滑动窗口的长度等于P的长度。

PS:这里开始我用49. 字母异位词分组同样的方法,利用排序找出p字符串和滑动窗口的key,再进行比较,发现力扣有些变态测试用例会严重超时。其他测试用例都可以通过。代码如下(仅供参考):

class Solution {
public:
    vector<int> findAnagrams(string s, string p) {
        vector<int> nums;
        sort(p.begin(),p.end());
        int left=0,right=p.size();
        while((left+right)<=s.size()){
            string result=s.substr(left,right);
            sort(result.begin(),result.end());
            if(result==p){
                nums.emplace_back(left);
            }
            ++left;
        }
        return nums;
    }
};

继续优化,耗时的原因就是排序,所以换成通过统计滑动窗口中字母出现的次数来判断是否和P是异位词。最后一个小细节,如果字符串s的长度小于字符串p的长度,直接返回空。

代码:

class Solution {
public:
    vector<int> findAnagrams(string s, string p) {
       int sL=s.size(),pL=p.size();
       if(sL<pL){//匹配字符串没有异位词长度大
        return vector<int>();
       }
       vector<int> nums;
       vector<int> snums(26);
       vector<int> pnums(26);
        for (int i = 0; i < pL; ++i) {//统计滑动窗口刚开始,内部包含的字母数量,相当于初始化滑动窗口
            ++snums[s[i] - 'a'];
            ++pnums[p[i] - 'a'];
        }

        if (snums == pnums) {
            nums.emplace_back(0);//如果滑动窗口启动时就匹配,说明位置0就是一个答案
        }
        for (int i = 0; i < sL - pL; ++i) {
            --snums[s[i] - 'a'];//滑动窗口左侧抛出
            ++snums[s[i + pL] - 'a'];//滑动窗口右侧加入

            if (snums == pnums) {
                nums.emplace_back(i + 1);//注意i位置是被抛出滑动窗口的,匹配成功应该记录i的下一个位置
            }
        }
        return nums;
    }
};

560. 和为 K 的子数组

给你一个整数数组 nums 和一个整数 k ,请你统计并返回 该数组中和为 k 的子数组的个数 

子数组是数组中元素的连续非空序列。

示例 1:

输入:nums = [1,1,1], k = 2
输出:2

示例 2:

输入:nums = [1,2,3], k = 3
输出:2

思路:注意这里子数组是连续的序列,所以不存在跳着组合加起来等于k。看到连续数组和就用前缀和去处理(前缀和不在这里介绍)。

前缀和数组中有两种情况,一是本身就等于k,二是存在两个前缀和相减等于k。

对于上述两种情况,需要用map数组去统计前缀和出现的次数

为什么呢?

首先对于情况一,直接输出等于k的前缀和出现的次数,所以遍历前缀和时用map去统计出现次数;

对于情况二,我们举一个例子,如下 

可以看到对于情况二,如果前缀和出现相等时,那么符合条件的子数组个数就等于该前缀和出现的次数。例如前缀和0出现次数等于2,所以符合结果的子数组次数也是2,到这里就可以发现为什么要用map统计前缀和出现的次数,用代码表示就是

sum//当前前缀和
sum-k//符合条件的前缀和
map.count(sum-k)//map中符合条件的前缀和是否出现
res+=map[sum-k];//统计map中符合条件的前缀和出现次数

最后整合情况一和情况二,对于情况一来说前缀和sum等于K,所以我们初始化时候,map[0]=1,这样只要符合前缀和出现时候,map[sum-k]就是map[0]出现次数,这样就将情况一和情况二整合了。

代码:

class Solution {
public:
    int subarraySum(vector<int>& nums, int k) {
        unordered_map<int,int> map;//存放前缀和
        int sum=0;//当前前缀和
        int res=0;
        map[0]=1;
        for(auto n:nums){
            sum+=n;//计算前缀和
            if(map.count(sum-k)){
                res+=map[sum-k];
            }
            map[sum]++;//当前前缀和次数加一
        }
        return res;
    }
};

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

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

相关文章

C语言 选择控制结构(1) 了解选择结构 关系运算符讲解 基本逻辑判断演示

接下来 我们来说 选择控制结构 在生活中 我们也有很多需要分支结构的例子 比如: 计算两个整数的最大值 计算n个数的最大值&#xff0c;最小值 判断三角形三边能否构成三角形? 判断某年是否是闰年? 判断输入的英文字母是大写还是小写? 我们在程序开发中 需要根据某种条件 进…

重磅!Meta 发布 Llama 3,前所未有的强大功能和多模态能力|TodayAI

Meta今日宣布推出其最新一代尖端开源大型语言模型Llama 3。该模型预计很快将在多个领先的云服务平台上线&#xff0c;包括AWS、Databricks、Google Cloud、Hugging Face、Kaggle、IBM WatsonX、Microsoft Azure、NVIDIA NIM和Snowflake。 Llama 3模型得到了AMD、AWS、Dell、In…

解决VirtualBox虚拟机启动失败的问题

一.出现的问题&#xff08;未能启动虚拟电脑&#xff0c;由于物理网卡未找到&#xff09; 一、错误信息分析 “未能启动虚拟电脑&#xff0c;由于物理网卡未找到”&#xff1a;这个错误通常是由于VirtualBox无法识别或连接到物理网卡造成的。可能是由于驱动程序问题、网络设置错…

Hadoop——Yarn 调度器和调度算法

Yarn 调度器和调度算法 YARN调度器&#xff08;Scheduler&#xff09;是负责将集群资源分配给不同应用程序的组件。它根据应用程序的资源需求和优先级&#xff0c;以及集群的资源供给情况&#xff0c;决定如何分配资源。YARN提供了多种调度器实现&#xff0c;每种调度器都有不…

力扣:219. 存在重复元素 II

力扣&#xff1a;219. 存在重复元素 II 给你一个整数数组 nums 和一个整数 k &#xff0c;判断数组中是否存在两个 不同的索引 i 和 j &#xff0c;满足 nums[i] nums[j] 且 abs(i - j) < k 。如果存在&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 …

服务调用-微服务小白入门(4)

背景 各个服务应用&#xff0c;有很多restful api&#xff0c;不论是用哪种方式发布&#xff0c;部署&#xff0c;注册&#xff0c;发现&#xff0c;有很多场景需要各个微服务之间进行服务的调用&#xff0c;大多时候返回的json格式响应数据多&#xff0c;如果是前端直接调用倒…

ST-GCN模型详解(+openpose)

ST-GCN模型详解&#xff08;openpose&#xff09; 一、什么是ST-GCN呢 基于骨架的动作识别&#xff08;Skeleton-Based Action Recognition&#xff09;主要任务是从一系列时间连续的骨骼关键点&#xff08;2D/3D&#xff09;中识别出正在执行的动作。因为牵涉到骨骼框架这种…

工控CTF之协议分析类型

协议分析 主要以工控流量和恶意流量为主&#xff0c;难度较低的题目主要考察Wireshark使用和找规律&#xff0c;难度较高的题目主要考察协议定义和特征 简单只能简单得干篇一律&#xff0c;难可以难得五花八门 常见的工控协议有&#xff1a;Modbus、MMS、IEC60870、MQTT、CoA…

完整版软件建模复习题和答案

一、单选题 D &#xff09;1&#xff0e;下面哪个不是信息系统利益相关者&#xff1f; A&#xff0e;客户 B&#xff0e;用户 C&#xff0e;开发人员 D&#xff0e;监理人员 B &#xff09;2&#xff0e;下面哪项不是用户主要关注的软件质量属性&#xff1f; A&#xff0e;…

解线性方程组——上三角、下三角,回代算法 | 北太天元

解上三角(回代) a i i ≠ 0 a_{ii\neq0} aii0​ , i 1 , 2 , … , n i1,2,\ldots,n i1,2,…,n a 11 x 1 a 12 x 2 ⋯ a 1 n x n b 1 a 22 x 2 ⋯ a 2 n x n b 2 ⋯ a n n x n b n \begin{aligned} a_{11}x_1a_{12}x_2\cdotsa_{1n}x_n&b_1 \\ a_{22}x_2\cdotsa_…

基于Matlab机器人工具箱对Dobot机械臂的研究

文章目录 文章目录 前言 一、Dobot Mangician 分析 二、Matlab 机器人工具箱 1. 建立模型 2. DoBot 正向运动学 3. Dobot 逆运动学 4. Dobot workpace 5. Dobot轨迹规划 三、Dobot studio 1. DoBot teaching 2. DoBot Python 程序 总结 前言 在本实验中&#xf…

自如电费均摊问题

3月份搬了次家&#xff0c;嫌麻烦租了自如&#xff0c;第一个月的电费账单出来了&#xff0c;由于我是中途搬进去的&#xff0c;于是乎就好奇他会如何计算均摊&#xff0c;这个月电费账单出来了&#xff0c;算了下发现了点东西。 先说结论&#xff1a;按照我的这个均摊的方式&a…

TCP报文与三次握手四次断开、TCP最大连接数与文件打开数限制、keepalive、tcpdump、wireshark抓包分析工具

TCP报文 tcp详解、tcp与udp对比等 TCP:传输控制协议 UDP&#xff1a;用户数据报协议 源端口和目的端口字段&#xff1a;各占 2 字节&#xff08;16位&#xff09;。端口是运输层与应用层的服务接口。运输层的复用和分用功能都要通过端口才能实现。 序列号&#xff1a;在建立…

万兆以太网10G Ethernet简介

2002年6月IEEE标准协会批准了万兆&#xff08;10G&#xff09;以太网的正式标准。此标准的全名是“10Gbit/s工作的媒体接入控制参数、物理层和管理参数”。 另一个组织是10G以太网联盟(10GEA)。10GEA由网络界的著名企业创建&#xff0c;现已有一百多家企业参加&#xff0c;中国…

Pytorch DistributedDataParallel(DDP)教程一:快速入门理论篇

Pytorch DistributedDataParallel&#xff08;DDP&#xff09;教程一&#xff1a;快速入门理论篇 目录 一、 写在前面二、什么是分布式并行训练1. 并行训练2. 数据并行 三、DDP的基本原理1. DDP的训练过程2. Ring-All-Reduce算法 四、如何搭建一个Pytorch DDP代码框架1. 与DDP有…

javaScript常用知识点

1. this指向问题 在绝大多数情况下&#xff0c;函数的调用方式决定了this的值。this不能在执行期间被赋值&#xff0c;并且在每次函数被调用时this的值也可能会不同。 this指向的对象称为函数的上下文对象context&#xff1b;this的指向取决于函数被调用方式this的指向不是函数…

【机器学习】小波变换在特征提取中的实践与应用

小波变换在特征提取中的实践与应用 一、小波变换的基本原理与数学表达二、基于小波变换的特征提取方法与实例三、小波变换在特征提取中的优势与展望 在信号处理与数据分析领域&#xff0c;小波变换作为一种强大的数学工具&#xff0c;其多尺度分析特性使得它在特征提取中扮演着…

云服务器部署Springboot项目

前端项目打包 修改ip地址 在控制台输入npm run build:prod 会产生dist文件 将dist文件中的内容移动至/usr/local/nginx/html目录下 后端项目打包 修改ip地址 执行clean操作 执行install操作 将生成的target文件中的jar包移动至/usr/local/src目录下 启动 注意⚠️&#xff…

【linux】Ubuntu 修改用户名

第一次打开Ubuntu时不小心把初始用户名“siriusiot”写成“siriousiot”&#xff08;多了一个o&#xff09; 。作为技术人&#xff0c;我们要保持严谨&#xff0c;我们要纠正过来&#xff08;其实就是单词拼错了怕被笑话&#xff09;。 打开终端&#xff0c;输入&#xff1a; …

Redis key(BigKey、MoreKey)的存储策略

1. MoreKey 案例 1.1 大批量往 redis 里面 插入2000w 测试数据key (1) Linux Bash 下面执行&#xff0c;插入 100w rootspray:~# for((i1;i<100*10000;i)); do echo "set k$i v$i" >> /tmp/redisTest.txt; done; 查看 rootspray:~# more /tmp/redisTest.…