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

2024-05-11 17:38

本文主要是介绍LeetCode - 438. 找到字符串中所有字母异位词,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

描述

给定一个字符串 s 和一个非空字符串 p,找到 s 中所有是 p 的字母异位词的子串,返回这些子串的起始索引。

字符串只包含小写英文字母,并且字符串 s 和 p 的长度都不超过 20100。

说明:

字母异位词指字母相同,但排列不同的字符串。
不考虑答案输出的顺序。
示例 1:

输入:
s: "cbaebabacd" p: "abc"

输出:
[0, 6]

解释:
起始索引等于 0 的子串是 "cba", 它是 "abc" 的字母异位词。
起始索引等于 6 的子串是 "bac", 它是 "abc" 的字母异位词。
 示例 2:

输入:
s: "abab" p: "ab"

输出:
[0, 1, 2]

解释:
起始索引等于 0 的子串是 "ab", 它是 "ab" 的字母异位词。
起始索引等于 1 的子串是 "ba", 它是 "ab" 的字母异位词。
起始索引等于 2 的子串是 "ab", 它是 "ab" 的字母异位词。

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/find-all-anagrams-in-a-string/
 

求解

    class Solution {public:// 方法一,暴力解法,以每个字符起点开始判断, 时间复杂度O(m * n),需要额外的空间复杂度// 低效率通过, 1300msvector<int> findAnagrams_1e(string s, string p) {const int n = s.size();const int m = p.size();vector<int> res;for (int i = 0; i <= n - m; ++i) {if (isAnagrams(p, s, i, i + m)) {res.emplace_back(i);}}return res;}// 方法二,暴力解法优化,以每个字符起点开始判断, 时间复杂度O(m * n),需要额外的空间复杂度// 低效率通过, 850msvector<int> findAnagrams_2e(string s, string p) {const int n = s.size();const int m = p.size();vector<int> res;bool preIsAnagrams = false;for (int i = 0; i <= n - m; ++i) {if (preIsAnagrams && s[i - 1] == s[i + m - 1]) {// 如果发现前一个满足条件,即s[i-1....i-1+m)符合条件// 则只需要判断s[i-1] 是否等于 s[i+m-1],即递增中去掉的那个字符和新加入的字符是否一致,一致则同样满足条件res.emplace_back(i);preIsAnagrams = true;continue;}if (isAnagrams(p, s, i, i + m)) {res.emplace_back(i);preIsAnagrams = true;continue;}preIsAnagrams = false;}return res;}// 方法三,双指针滑动窗口vector<int> findAnagrams(string s, string p) {const int n = s.size();const int m = p.size();int record[126]{0};for (char &c : p) {++record[c];}int l = 0, r = 0, num = 0;vector<int> res;while (r < n) {if (record[s[r]] > 0) {++num;}--record[s[r]];++r;while (num == m) {if (r - l == m) {res.emplace_back(l);}if (record[s[l]] == 0) {--num;}++record[s[l]];++l;}}return res;}private:/*!** @param p 被匹配字符串* @param s 待匹配字符串* @param is 起点,相当于begin* @param ie 终点, 相等于end* @return 返回是否是异位匹配*/bool isAnagrams(const std::string_view p, const std::string_view s, int is, int ie) {int pattern[26]{0};for (const auto &c : p) {++pattern[c - 'a'];}for (int i = is; i < ie; ++i) {--pattern[s[i] - 'a'];}for (const int &num : pattern) {if (num != 0) {return false;}}return true;}};

 

这篇关于LeetCode - 438. 找到字符串中所有字母异位词的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/980205

相关文章

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码

《在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码》在MyBatis的XML映射文件中,trim元素用于动态添加SQL语句的一部分,处理前缀、后缀及多余的逗号或连接符,示... 在MyBATis的XML映射文件中,<trim>元素用于动态地添加SQL语句的一部分,例如SET或W

C#实现获得某个枚举的所有名称

《C#实现获得某个枚举的所有名称》这篇文章主要为大家详细介绍了C#如何实现获得某个枚举的所有名称,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... C#中获得某个枚举的所有名称using System;using System.Collections.Generic;usi

通过C#获取PDF中指定文本或所有文本的字体信息

《通过C#获取PDF中指定文本或所有文本的字体信息》在设计和出版行业中,字体的选择和使用对最终作品的质量有着重要影响,然而,有时我们可能会遇到包含未知字体的PDF文件,这使得我们无法准确地复制或修改文... 目录引言C# 获取PDF中指定文本的字体信息C# 获取PDF文档中用到的所有字体信息引言在设计和出

python修改字符串值的三种方法

《python修改字符串值的三种方法》本文主要介绍了python修改字符串值的三种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录第一种方法:第二种方法:第三种方法:在python中,字符串对象是不可变类型,所以我们没办法直接

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

C#中字符串分割的多种方式

《C#中字符串分割的多种方式》在C#编程语言中,字符串处理是日常开发中不可或缺的一部分,字符串分割是处理文本数据时常用的操作,它允许我们将一个长字符串分解成多个子字符串,本文给大家介绍了C#中字符串分... 目录1. 使用 string.Split2. 使用正则表达式 (Regex.Split)3. 使用

Java中JSON字符串反序列化(动态泛型)

《Java中JSON字符串反序列化(动态泛型)》文章讨论了在定时任务中使用反射调用目标对象时处理动态参数的问题,通过将方法参数存储为JSON字符串并进行反序列化,可以实现动态调用,然而,这种方式容易导... 需求:定时任务扫描,反射调用目标对象,但是,方法的传参不是固定的。方案一:将方法参数存成jsON字

哈希leetcode-1

目录 1前言 2.例题  2.1两数之和 2.2判断是否互为字符重排 2.3存在重复元素1 2.4存在重复元素2 2.5字母异位词分组 1前言 哈希表主要是适合于快速查找某个元素(O(1)) 当我们要频繁的查找某个元素,第一哈希表O(1),第二,二分O(log n) 一般可以分为语言自带的容器哈希和用数组模拟的简易哈希。 最简单的比如数组模拟字符存储,只要开26个c

usaco 1.2 Name That Number(数字字母转化)

巧妙的利用code[b[0]-'A'] 将字符ABC...Z转换为数字 需要注意的是重新开一个数组 c [ ] 存储字符串 应人为的在末尾附上 ‘ \ 0 ’ 详见代码: /*ID: who jayLANG: C++TASK: namenum*/#include<stdio.h>#include<string.h>int main(){FILE *fin = fopen (