欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C C++算法題解LeetCode1408數(shù)組中的字符串匹配

 更新時(shí)間:2022年10月14日 09:21:10   作者:Junkman丶  
這篇文章主要為大家介紹了C C++算法題解LeetCode1408數(shù)組中的字符串匹配示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

題目描述

題目鏈接:1408. 數(shù)組中的字符串匹配

給你一個(gè)字符串?dāng)?shù)組 words ,數(shù)組中的每個(gè)字符串都可以看作是一個(gè)單詞。請(qǐng)你按 任意 順序返回 words 中是其他單詞的子字符串的所有單詞。

如果你可以刪除 words[j] 最左側(cè)和/或最右側(cè)的若干字符得到 word[i] ,那么字符串 words[i] 就是 words[j] 的一個(gè)子字符串。

提示:

示例 1:

輸入:words = ["mass","as","hero","superhero"]
輸出:["as","hero"]
解釋?zhuān)?quot;as" 是 "mass" 的子字符串,"hero" 是 "superhero" 的子字符串。
["hero","as"] 也是有效的答案。

示例 2:

輸入:words = ["leetcode","et","code"]
輸出:["et","code"]
解釋?zhuān)?quot;et" 和 "code" 都是 "leetcode" 的子字符串。

示例 3:

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

整理題意

題目給定一個(gè)字符串?dāng)?shù)組 words,對(duì)于數(shù)組中的每個(gè)字符串來(lái)說(shuō),如果該字符串為數(shù)組中其他某個(gè)字符串的子串,那么就將該字符串加入答案字符串?dāng)?shù)組??梢园凑杖我忭樞蚍祷卦摯鸢笖?shù)組。

解題思路分析

注意題目的數(shù)據(jù)提示:題目數(shù)據(jù) 保證 每個(gè) words[i] 都是獨(dú)一無(wú)二的。所以不存在兩個(gè)相同的字符串,也避免了互為子字符串的情況。

根據(jù)題目數(shù)據(jù)范圍來(lái)看,完全可以采用較為暴力的方法來(lái)進(jìn)行解題,枚舉每個(gè)字符串作為子串,檢查是否為其他某個(gè)字符串的子串即可。

優(yōu)化

在字符串匹配的時(shí)候可以采用 KMP 字符串匹配算法來(lái)進(jìn)行優(yōu)化時(shí)間復(fù)雜度。

具體實(shí)現(xiàn)

對(duì)于字符串匹配部分可以調(diào)用 string 中的 find() 函數(shù)進(jìn)行匹配 t.find(p)(在字符串 t 中匹配字符串 p,也就是查找字符串 t 中是否包含字符串 p):

  • 此處需要用到 string 庫(kù)中的 find() 函數(shù)與 string::npos 參數(shù);

string::npos 參數(shù)是一個(gè)常數(shù),用來(lái)表示不存在的位置。

  • stringfind() 返回值是子串的第一個(gè)字符在母串中的位置(下標(biāo)記錄),如果沒(méi)有找到,那么會(huì)返回一個(gè)特別的標(biāo)記 string::npos。

可以對(duì)字符串?dāng)?shù)組 words 進(jìn)行排序處理,這樣就可以從最短的字符串開(kāi)始匹配,且每次往后遍歷匹配,因?yàn)榍懊娴淖址欢ǘ逃诋?dāng)前字符串。

在使用 KMP 字符串匹配算法時(shí)需要注意:

  • KMP 字符串匹配算法的核心思想是 遞歸回溯思想,當(dāng)匹配失敗時(shí)根據(jù) nxt 數(shù)組來(lái)進(jìn)行回溯跳轉(zhuǎn);
  • nxt 數(shù)組表示模式串的子串的前綴和后綴相同的最長(zhǎng)長(zhǎng)度,這樣就可以在匹配的過(guò)程中如果遇到不匹配的字符,模式串用 nxt 數(shù)組進(jìn)行遞歸跳轉(zhuǎn)到最長(zhǎng)符合的位置進(jìn)行繼續(xù)匹配,從而不需要目標(biāo)串進(jìn)行重復(fù)的往返匹配。
  • 其中需要要注意的一個(gè)技巧是 nxt[0] = -1,在把 nxt 數(shù)組進(jìn)行向右偏移時(shí),第 0 位的值,我們將其設(shè)成了 -1,這只是為了編程的方便,并沒(méi)有其他的意義。
  • 還需要注意 nxt 數(shù)組的優(yōu)化,優(yōu)化后在回溯跳轉(zhuǎn)的時(shí)候會(huì)回溯跳轉(zhuǎn)到首次與當(dāng)前字符不一樣字符的位置,避免了跳轉(zhuǎn)到和當(dāng)前字符一樣的位置進(jìn)行重復(fù)判斷。
  • 在實(shí)現(xiàn) getNext() 函數(shù)的時(shí)候需要注意 nxt 數(shù)組溢出問(wèn)題,可以通過(guò)增加 nxt 數(shù)組大小,或減少 getNext() 函數(shù)中循環(huán)遍歷的次數(shù)來(lái)防止越界出現(xiàn)的運(yùn)行錯(cuò)誤。
  • 需要注意在 getNext() 函數(shù)中 j 的初始化為 -1,但在 KMP() 函數(shù)中 j 的初始化為 0。

復(fù)雜度分析

代碼實(shí)現(xiàn)

暴力

class Solution {
public:
    vector<string> stringMatching(vector<string>& words) {
        // 新知識(shí):string::npos
        vector<string> ans;
        ans.clear();
        // 雙重循環(huán)暴力尋找
        for(auto &word1 : words){
            int l1 = word1.length();
            for(auto &word2 : words){
                int l2 = word2.length();
                // 當(dāng) l2 大于 l1 時(shí) 并且可以在 w2 中找到 w1 時(shí)
                if(l1 < l2 && word2.find(word1) != string::npos){
                    ans.emplace_back(word1);
                    break;
                }
            }
        }
        return ans;
    }
};

暴力 + 優(yōu)化

class Solution {
public:
    vector<string> stringMatching(vector<string>& words) {
        sort(words.begin(), words.end(), [](string &a, string &b){
            return a.length() < b.length();
        });
        // 新知識(shí):string::npos
        vector<string> ans;
        ans.clear();
        int n = words.size();
        // 雙重循環(huán)暴力尋找
        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();
                // 當(dāng) l2 大于 l1 時(shí) 并且可以在 w2 中找到 w1 時(shí)
                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進(jìn)行向右偏移時(shí),第0位的值,我們將其設(shè)成了-1,
        // 這只是為了編程的方便,并沒(méi)有其他的意義。
        nxt[0] = -1;
        int i = 0, j = -1;
        int len = p.length();
        // ★注意 nxt 數(shù)組越界
        while(i < len){
            // j = -1 或者 匹配成功
            if(j == -1 || p[i] == p[j]){
                // nxt[++i] = ++j; 未優(yōu)化前
                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 數(shù)組溢出
            // 可以這里 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;
    }
};

總結(jié)

  • 通過(guò)該題了解到了一個(gè)新的知識(shí)點(diǎn):string::npos 參數(shù)用來(lái)表示不存在的位置。當(dāng) stringfind() 函數(shù)沒(méi)有匹配成功時(shí),那么就會(huì)返回這個(gè)參數(shù) string::npos。
  • 同時(shí)通過(guò)該題復(fù)習(xí)了 KMP 字符串匹配算法 的實(shí)現(xiàn),在實(shí)現(xiàn)過(guò)程中需要注意 nxt 數(shù)組的大小,防止下標(biāo)越界的運(yùn)行錯(cuò)誤;同時(shí)還需要注意在 getNext() 函數(shù)中 j 的初始化為 -1,但在 KMP() 函數(shù)中 j 的初始化為 0。

測(cè)試結(jié)果:

以上就是C C++算法題解LeetCode1408數(shù)組中的字符串匹配的詳細(xì)內(nèi)容,更多關(guān)于C C++算法數(shù)組字符串匹配的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • C++ Qt開(kāi)發(fā)之使用QHostInfo查詢(xún)主機(jī)地址

    C++ Qt開(kāi)發(fā)之使用QHostInfo查詢(xún)主機(jī)地址

    Qt 是一個(gè)跨平臺(tái)C++圖形界面開(kāi)發(fā)庫(kù),利用Qt可以快速開(kāi)發(fā)跨平臺(tái)窗體應(yīng)用程序,本文將重點(diǎn)介紹如何運(yùn)用QHostInfo組件實(shí)現(xiàn)對(duì)主機(jī)地址查詢(xún)功能,希望對(duì)大家有所幫助
    2024-03-03
  • C++程序中main(int argc, char *argv[])函數(shù)的參數(shù)意義

    C++程序中main(int argc, char *argv[])函數(shù)的參數(shù)意義

    這篇文章主要介紹了C++程序中main(int argc, char *argv[])函數(shù)的參數(shù)意義,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2018-09-09
  • C語(yǔ)言實(shí)現(xiàn)BMP轉(zhuǎn)換JPG的方法

    C語(yǔ)言實(shí)現(xiàn)BMP轉(zhuǎn)換JPG的方法

    這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)BMP轉(zhuǎn)換JPG的方法,涉及C#圖片格式轉(zhuǎn)換的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-07-07
  • 詳解C++11原子類(lèi)型與原子操作

    詳解C++11原子類(lèi)型與原子操作

    這篇文章主要介紹了C++11原子類(lèi)型與原子操作的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)c++,感興趣的朋友可以了解下
    2020-08-08
  • C語(yǔ)言編程計(jì)算信噪比SNR理解學(xué)習(xí)

    C語(yǔ)言編程計(jì)算信噪比SNR理解學(xué)習(xí)

    這篇文章主要介紹了C語(yǔ)言編程信噪比SNR計(jì)算的理解學(xué)習(xí),信噪比,英文名稱(chēng)叫做SNR或S/N(SIGNAL-NOISE RATIO)。是指一個(gè)電子設(shè)備或者電子系統(tǒng)中信號(hào)與噪聲的比例
    2021-10-10
  • C++深入講解哈夫曼樹(shù)

    C++深入講解哈夫曼樹(shù)

    給定N個(gè)權(quán)值作為N個(gè)葉子結(jié)點(diǎn),構(gòu)造一棵二叉樹(shù),若該樹(shù)的帶權(quán)路徑長(zhǎng)度達(dá)到最小,稱(chēng)這樣的二叉樹(shù)為最優(yōu)二叉樹(shù),也稱(chēng)為哈夫曼樹(shù)(Huffman Tree)。哈夫曼樹(shù)是帶權(quán)路徑長(zhǎng)度最短的樹(shù),權(quán)值較大的結(jié)點(diǎn)離根較近
    2022-05-05
  • C語(yǔ)言菜鳥(niǎo)基礎(chǔ)教程之常量和變量

    C語(yǔ)言菜鳥(niǎo)基礎(chǔ)教程之常量和變量

    在C語(yǔ)言中,常量和變量都是可以用來(lái)存儲(chǔ)和表示數(shù)據(jù)的,常量值在程序執(zhí)行的過(guò)程中是不可變的,而變量是可變的
    2017-10-10
  • QT窗口/控件置頂方法舉例詳解

    QT窗口/控件置頂方法舉例詳解

    我們使用QT進(jìn)行界面開(kāi)發(fā)時(shí),可能會(huì)遇到需要將窗口置頂?shù)那闆r,下面這篇文章主要給大家介紹了關(guān)于QT窗口/控件置頂方法的相關(guān)資料,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2024-01-01
  • opencv利用視頻的前n幀求平均圖像

    opencv利用視頻的前n幀求平均圖像

    這篇文章主要為大家詳細(xì)介紹了opencv利用視頻的前n幀求平均圖像,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-03-03
  • C++基本用法實(shí)踐之移動(dòng)語(yǔ)義詳解

    C++基本用法實(shí)踐之移動(dòng)語(yǔ)義詳解

    移動(dòng)(move)語(yǔ)義是C++引入了一種新的內(nèi)存優(yōu)化,以避免不必要的拷貝,下面小編就來(lái)和大家簡(jiǎn)單聊聊C++中移動(dòng)語(yǔ)義的相關(guān)使用吧,希望對(duì)大家有所幫助
    2023-07-07

最新評(píng)論