返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >CC++算法题解LeetCode1408数组中的字符串匹配
  • 554
分享到

CC++算法题解LeetCode1408数组中的字符串匹配

CC++算法数组字符串匹配CC++数组字符串匹配 2022-11-13 18:11:52 554人浏览 泡泡鱼
摘要

目录题目描述整理题意解题思路分析优化具体实现复杂度分析代码实现暴力暴力 + 优化KMP总结题目描述 题目链接:1408. 数组中的字符串匹配 给你一个字符串数组 Words ,数组中

题目描述

题目链接:1408. 数组中的字符串匹配

给你一个字符串数组 Words ,数组中的每个字符串都可以看作是一个单词。请你按 任意 顺序返回 words 中是其他单词的子字符串的所有单词。

如果你可以删除 words[j] 最左侧和/或最右侧的若干字符得到 word[i] ,那么字符串 words[i] 就是 words[j] 的一个子字符串。

提示:

示例 1:

输入:words = ["mass","as","hero","superhero"]
输出:["as","hero"]
解释:"as" 是 "mass" 的子字符串,"hero" 是 "superhero" 的子字符串。
["hero","as"] 也是有效的答案。

示例 2:

输入:words = ["LeetCode","et","code"]
输出:["et","code"]
解释:"et" 和 "code" 都是 "leetcode" 的子字符串。

示例 3:

输入: words = ["blue","green","bu"]
输出: []

整理题意

题目给定一个字符串数组 words,对于数组中的每个字符串来说,如果该字符串为数组中其他某个字符串的子串,那么就将该字符串加入答案字符串数组。可以按照任意顺序返回该答案数组。

解题思路分析

注意题目的数据提示:题目数据 保证 每个 words[i] 都是独一无二的。所以不存在两个相同的字符串,也避免了互为子字符串的情况。

根据题目数据范围来看,完全可以采用较为暴力的方法来进行解题,枚举每个字符串作为子串,检查是否为其他某个字符串的子串即可。

优化

在字符串匹配的时候可以采用 KMP 字符串匹配算法来进行优化时间复杂度。

具体实现

对于字符串匹配部分可以调用 string 中的 find() 函数进行匹配 t.find(p)(在字符串 t 中匹配字符串 p,也就是查找字符串 t 中是否包含字符串 p):

  • 此处需要用到 string 库中的 find() 函数与 string::npos 参数;

string::npos 参数是一个常数,用来表示不存在的位置。

  • stringfind() 返回值是子串的第一个字符在母串中的位置(下标记录),如果没有找到,那么会返回一个特别的标记 string::npos

可以对字符串数组 words 进行排序处理,这样就可以从最短的字符串开始匹配,且每次往后遍历匹配,因为前面的字符串一定短于当前字符串。

在使用 KMP 字符串匹配算法时需要注意:

  • KMP 字符串匹配算法的核心思想是 递归回溯思想,当匹配失败时根据 nxt 数组来进行回溯跳转;
  • nxt 数组表示模式串的子串的前缀和后缀相同的最长长度,这样就可以在匹配的过程中如果遇到不匹配的字符,模式串用 nxt 数组进行递归跳转到最长符合的位置进行继续匹配,从而不需要目标串进行重复的往返匹配。
  • 其中需要要注意的一个技巧是 nxt[0] = -1,在把 nxt 数组进行向右偏移时,第 0 位的值,我们将其设成了 -1,这只是为了编程的方便,并没有其他的意义。
  • 还需要注意 nxt 数组的优化,优化后在回溯跳转的时候会回溯跳转到首次与当前字符不一样字符的位置,避免了跳转到和当前字符一样的位置进行重复判断。
  • 在实现 getNext() 函数的时候需要注意 nxt 数组溢出问题,可以通过增加 nxt 数组大小,或减少 getNext() 函数中循环遍历的次数来防止越界出现的运行错误。
  • 需要注意在 getNext() 函数中 j 的初始化为 -1,但在 KMP() 函数中 j 的初始化为 0

复杂度分析

代码实现

暴力

class Solution {
public:
    vector<string> stringMatching(vector<string>& words) {
        // 新知识:string::npos
        vector<string> ans;
        ans.clear();
        // 双重循环暴力寻找
        for(auto &word1 : words){
            int l1 = word1.length();
            for(auto &word2 : words){
                int l2 = word2.length();
                // 当 l2 大于 l1 时 并且可以在 w2 中找到 w1 时
                if(l1 < l2 && word2.find(word1) != string::npos){
                    ans.emplace_back(word1);
                    break;
                }
            }
        }
        return ans;
    }
};

暴力 + 优化

class Solution {
public:
    vector<string> stringMatching(vector<string>& words) {
        sort(words.begin(), words.end(), [](string &a, string &b){
            return a.length() < b.length();
        });
        // 新知识:string::npos
        vector<string> ans;
        ans.clear();
        int n = words.size();
        // 双重循环暴力寻找
        for(int i = 0; i < n; i++){
            int l1 = words[i].length();
            for(int j = i + 1; j < n; j++){
                int l2 = words[j].length();
                // 当 l2 大于 l1 时 并且可以在 w2 中找到 w1 时
                if(l1 < l2 && words[j].find(words[i]) != string::npos){
                    ans.emplace_back(words[i]);
                    break;
                }
            }
        }
        return ans;
    }
};

KMP

class Solution {
    void getNext(string &p, vector<int> &nxt){
        // 把PMT进行向右偏移时,第0位的值,我们将其设成了-1,
        // 这只是为了编程的方便,并没有其他的意义。
        nxt[0] = -1;
        int i = 0, j = -1;
        int len = p.length();
        // ★注意 nxt 数组越界
        while(i < len){
            // j = -1 或者 匹配成功
            if(j == -1 || p[i] == p[j]){
                // nxt[++i] = ++j; 未优化前
                i++;
                j++;
                if(p[i] == p[j]) nxt[i] = nxt[j];
                else nxt[i] = j;
            }
            // 匹配失败,回溯
            else{
                j = nxt[j];
            }
        }
    }
    bool kmp(string &t, string &p, vector<int> &nxt){
        // ★注意这里的 j = 0 不是 j = -1
        int i = 0, j = 0;
        int lent = t.length();
        int lenp = p.length();
        while(i < lent && j < lenp){
            if(j == -1 || t[i] == p[j]){
                ++i;
                ++j;
            }
            else j = nxt[j];
        }
        if(j == lenp) return true;
        return false;
    }
public:
    vector<string> stringMatching(vector<string>& words) {
        sort(words.begin(), words.end(), [](string a, string b){
            return a.length() < b.length();
        });
        vector<string> ans;
        ans.clear();
        vector<int> nxt;
        int n = words.size();
        for(int i = 0; i < n; i++){
            int len_p = words[i].length();
            // ★注意 nxt 数组溢出
            // 可以这里 len_p + 1 也可以 getNext 中 -1
            nxt.resize(len_p + 1);
            getNext(words[i], nxt);
            for(int j = i + 1; j < n; j++){
                if(kmp(words[j], words[i], nxt)){
                    ans.emplace_back(words[i]);
                    break;
                }
            }
        }
        return ans;
    }
};

总结

  • 通过该题了解到了一个新的知识点:string::npos 参数用来表示不存在的位置。当 stringfind() 函数没有匹配成功时,那么就会返回这个参数 string::npos
  • 同时通过该题复习了 KMP 字符串匹配算法 的实现,在实现过程中需要注意 nxt 数组的大小,防止下标越界的运行错误;同时还需要注意在 getNext() 函数中 j 的初始化为 -1,但在 KMP() 函数中 j 的初始化为 0

测试结果:

以上就是C c++算法题解LeetCode1408数组中的字符串匹配的详细内容,更多关于C C++算法数组字符串匹配的资料请关注编程网其它相关文章!

--结束END--

本文标题: CC++算法题解LeetCode1408数组中的字符串匹配

本文链接: https://lsjlt.com/news/169374.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • CC++算法题解LeetCode1408数组中的字符串匹配
    目录题目描述整理题意解题思路分析优化具体实现复杂度分析代码实现暴力暴力 + 优化KMP总结题目描述 题目链接:1408. 数组中的字符串匹配 给你一个字符串数组 words ,数组中...
    99+
    2022-11-13
    C C++算法数组字符串匹配 C C++ 数组字符串匹配
  • c++KMP字符串匹配算法
    目录KMP算法简介前缀表如何构造前缀表next数组如何用next数组进行模板匹配总结KMP算法简介        ...
    99+
    2024-04-02
  • java暴力匹配及KMP算法解决字符串匹配问题示例详解
    目录要解决的问题?一、暴力匹配算法一个图例介绍KMP算法二、KMP算法算法介绍一个图例介绍KMP算法  代码实现要解决的问题? 一、暴力匹配算法 一个图例介绍KMP算法 St...
    99+
    2024-04-02
  • 如何理解字符串匹配的Boyer-Moore算法
    这篇文章给大家介绍如何理解字符串匹配的Boyer-Moore算法,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。之前我介绍了KMP算法。但是,它并不是效率***的算法,实际采用并不多。各种文本编辑器的"查找&q...
    99+
    2023-06-17
  • 如何使用java暴力匹配及KMP算法解决字符串匹配问题
    这篇文章主要介绍如何使用java暴力匹配及KMP算法解决字符串匹配问题,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!要解决的问题?一、暴力匹配算法一个图例介绍KMP算法String str1 =&...
    99+
    2023-06-21
  • shell如何匹配字符串中的数字
    在shell中,可以使用正则表达式来匹配字符串中的数字。可以使用grep命令来进行匹配,具体的语法如下:```shellgrep -...
    99+
    2023-09-26
    shell
  • python怎么匹配字符串中的数字
    要匹配字符串中的数字,可以使用正则表达式来实现。下面是一个简单的示例,演示如何使用正则表达式来匹配字符串中的数字: import r...
    99+
    2024-04-08
    python
  • java字符串模糊匹配算法怎么应用
    字符串模糊匹配算法可以应用于各种场景,例如:1. 文本搜索引擎:在搜索引擎中,用户输入的查询字符串通常是模糊的,可以使用字符串模糊匹...
    99+
    2023-09-14
    java
  • shell字符串匹配的实现方法
    这篇文章主要介绍了shell字符串匹配的实现方法,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。一、简介      Bash ...
    99+
    2023-06-09
  • Bash编程中如何实现高效的字符串匹配算法?
    在Bash编程中,字符串匹配是一个非常常见的需求。无论是对文件名的匹配,还是对文本内容的搜索,字符串匹配都是必不可少的。然而,在实际编程中,我们经常会遇到一些字符串匹配效率较低的问题,这给程序的性能带来了很大的影响。本文将介绍Bash编程...
    99+
    2023-08-07
    编程算法 自然语言处理 bash
  • Python正则表达式匹配字符串中的数字
    这篇文章主要介绍了Python正则表达式匹配字符串中的数字,本文通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下1.使用“\d+”匹配全数字...
    99+
    2023-06-01
  • Java算法中数组与字符串练习题有哪些
    这篇文章主要介绍Java算法中数组与字符串练习题有哪些,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!题目一解法class Solution {    pub...
    99+
    2023-06-29
  • python怎么匹配字符串中间的文字
    你可以使用正则表达式来匹配字符串中间的文字。以下是一个例子:```pythonimport retext = "Hello Worl...
    99+
    2023-08-30
    python
  • sql如何匹配字符串中的某个字
    在SQL中,可以使用LIKE操作符来匹配字符串中的某个字。以下是一个示例: 假设有一个名为products的表,其中包含一个名为na...
    99+
    2024-04-15
    sql
  • PHP如何计算两个字符串的匹配度
    这篇文章主要讲解了“PHP如何计算两个字符串的匹配度”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“PHP如何计算两个字符串的匹配度”吧!计算两个字符串匹配度(相似度),也就是计算两个字符串的...
    99+
    2023-06-20
  • mysql字符串匹配的方法是什么
    MySQL提供了多种方法来进行字符串匹配,其中常用的有以下几种:1. LIKE操作符:LIKE操作符是最常用的字符串匹配方法,它可以...
    99+
    2023-10-09
    mysql
  • shell字符串匹配的方法有哪些
    使用通配符:通配符是一种简单而有效的方法来匹配字符串。在Shell中,可以使用星号(*)和问号()来代表零个或多个字符以及一个字...
    99+
    2024-04-02
  • MySQL中的字符串模式匹配实例讲解
    这篇文章主要介绍“MySQL中的字符串模式匹配实例讲解”,在日常操作中,相信很多人在MySQL中的字符串模式匹配实例讲解问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”MySQ...
    99+
    2024-04-02
  • Python做简单的字符串匹配详解
    Python做简单的字符串匹配详解 由于需要在半结构化的文本数据中提取一些特定格式的字段、数据辅助挖掘分析工作,以往都是使用Matlab工具进行结构化数据处理的建模,matlab擅长矩阵处理、结构化数据的...
    99+
    2022-06-04
    字符串 详解 简单
  • python怎么将字符串中匹配的数字乘于2
    本文小编为大家详细介绍“python怎么将字符串中匹配的数字乘于2”,内容详细,步骤清晰,细节处理妥当,希望这篇“python怎么将字符串中匹配的数字乘于2”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。repl ...
    99+
    2023-06-08
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作