国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - C/C++ - C++實現LeetCode(30.串聯所有單詞的子串)

C++實現LeetCode(30.串聯所有單詞的子串)

2021-11-25 15:29Grandyang C/C++

這篇文章主要介紹了C++實現LeetCode(30.串聯所有單詞的子串),本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下

[LeetCode] 30. Substring with Concatenation of All Words 串聯所有單詞的子串

You are given a string, s, and a list of words, words, that are all of the same length. Find all starting indices of substring(s) in s that is a concatenation of each word in words exactly once and without any intervening characters.

Example 1:

Input:
s = "barfoothefoobarman",
words = ["foo","bar"]
Output: [0,9]
Explanation: Substrings starting at index 0 and 9 are "barfoor" and "foobar" respectively.
The output order does not matter, returning [9,0] is fine too.

Example 2:

Input:
s = "wordgoodgoodgoodbestword",
words = ["word","good","best","word"]
Output: []

這道題讓我們求串聯所有單詞的子串,就是說給定一個長字符串,再給定幾個長度相同的單詞,讓找出串聯給定所有單詞的子串的起始位置,還是蠻有難度的一道題。假設 words 數組中有n個單詞,每個單詞的長度均為 len,那么實際上這道題就讓我們出所有長度為 nxlen 的子串,使得其剛好是由 words 數組中的所有單詞組成。那么就需要經常判斷s串中長度為 len 的子串是否是 words 中的單詞,為了快速的判斷,可以使用 HashMap,同時由于 words 數組可能有重復單詞,就要用 HashMap 來建立所有的單詞和其出現次數之間的映射,即統計每個單詞出現的次數。

遍歷s中所有長度為 nxlen 的子串,當剩余子串的長度小于 nxlen 時,就不用再判斷了。所以i從0開始,到 (int)s.size() - nxlen 結束就可以了,注意這里一定要將 s.size() 先轉為整型數,再進行解法。一定要形成這樣的習慣,一旦 size() 后面要減去數字時,先轉為 int 型,因為 size() 的返回值是無符號型,一旦減去一個比自己大的數字,則會出錯。對于每個遍歷到的長度為 nxlen 的子串,需要驗證其是否剛好由 words 中所有的單詞構成,檢查方法就是每次取長度為 len 的子串,看其是否是 words 中的單詞。為了方便比較,建立另一個 HashMap,當取出的單詞不在 words 中,直接 break 掉,否則就將其在新的 HashMap 中的映射值加1,還要檢測若其映射值超過原 HashMap 中的映射值,也 break 掉,因為就算當前單詞在 words 中,但若其出現的次數超過 words 中的次數,還是不合題意的。在 for 循環外面,若j正好等于n,說明檢測的n個長度為 len 的子串都是 words 中的單詞,并且剛好構成了 words,則將當前位置i加入結果 res 即可,具體參見代碼如下:

解法一:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
class Solution {
public:
    vector<int> findSubstring(string s, vector<string>& words) {
        if (s.empty() || words.empty()) return {};
        vector<int> res;
        int n = words.size(), len = words[0].size();
        unordered_map<string, int> wordCnt;
        for (auto &word : words) ++wordCnt[word];
        for (int i = 0; i <= (int)s.size() - n * len; ++i) {
            unordered_map<string, int> strCnt;
            int j = 0;
            for (j = 0; j < n; ++j) {
                string t = s.substr(i + j * len, len);
                if (!wordCnt.count(t)) break;
                ++strCnt[t];
                if (strCnt[t] > wordCnt[t]) break;
            }
            if (j == n) res.push_back(i);
        }
        return res;
    }
};

這道題還有一種 O(n) 時間復雜度的解法,設計思路非常巧妙,但是感覺很難想出來,博主目測還未到達這種水平。這種方法不再是一個字符一個字符的遍歷,而是一個詞一個詞的遍歷,比如根據題目中的例子,字符串s的長度n為 18,words 數組中有兩個單詞 (cnt=2),每個單詞的長度 len 均為3,那么遍歷的順序為 0,3,6,8,12,15,然后偏移一個字符 1,4,7,9,13,16,然后再偏移一個字符 2,5,8,10,14,17,這樣就可以把所有情況都遍歷到,還是先用一個 HashMap m1 來記錄 words 里的所有詞,然后從0開始遍歷,用 left 來記錄左邊界的位置,count 表示當前已經匹配的單詞的個數。然后一個單詞一個單詞的遍歷,如果當前遍歷的到的單詞t在 m1 中存在,那么將其加入另一個 HashMap m2 中,如果在 m2 中個數小于等于 m1 中的個數,那么 count 自增1,如果大于了,則需要做一些處理,比如下面這種情況:s = barfoofoo, words = {bar, foo, abc},給 words 中新加了一個 abc ,目的是為了遍歷到 barfoo 不會停止,當遍歷到第二 foo 的時候,  m2[foo]=2, 而此時 m1[foo]=1,這時候已經不連續了,所以要移動左邊界 left 的位置,先把第一個詞 t1=bar 取出來,然后將 m2[t1] 自減1,如果此時 m2[t1]<m1[t1] 了,說明一個匹配沒了,那么對應的 count 也要自減1,然后左邊界加上個 len,這樣就可以了。如果某個時刻 count 和 cnt 相等了,說明成功匹配了一個位置,將當前左邊界 left 存入結果 res 中,此時去掉最左邊的一個詞,同時 count 自減1,左邊界右移 len,繼續匹配。如果匹配到一個不在 m1 中的詞,說明跟前面已經斷開了,重置 m2,count 為0,左邊界 left 移到 j+len,參見代碼如下:

解法二:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
class Solution {
public:
    vector<int> findSubstring(string s, vector<string>& words) {
        if (s.empty() || words.empty()) return {};
        vector<int> res;
        int n = s.size(), cnt = words.size(), len = words[0].size();
        unordered_map<string, int> m1;
        for (string w : words) ++m1[w];
        for (int i = 0; i < len; ++i) {
            int left = i, count = 0;
            unordered_map<string, int> m2;
            for (int j = i; j <= n - len; j += len) {
                string t = s.substr(j, len);
                if (m1.count(t)) {
                    ++m2[t];
                    if (m2[t] <= m1[t]) {
                        ++count;
                    } else {
                        while (m2[t] > m1[t]) {
                            string t1 = s.substr(left, len);
                            --m2[t1];
                            if (m2[t1] < m1[t1]) --count;
                            left += len;
                        }
                    }
                    if (count == cnt) {
                        res.push_back(left);
                        --m2[s.substr(left, len)];
                        --count;
                        left += len;
                    }
                } else {
                    m2.clear();
                    count = 0;
                    left = j + len;
                }
            }
        }
        return res;
    }
};

到此這篇關于C++實現LeetCode(30.串聯所有單詞的子串)的文章就介紹到這了,更多相關C++實現串聯所有單詞的子串內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!

原文鏈接:https://www.cnblogs.com/grandyang/p/4521224.html

延伸 · 閱讀

精彩推薦
  • C/C++學習C++編程的必備軟件

    學習C++編程的必備軟件

    本文給大家分享的是作者在學習使用C++進行編程的時候所用到的一些常用的軟件,這里推薦給大家...

    謝恩銘10102021-05-08
  • C/C++深入理解goto語句的替代實現方式分析

    深入理解goto語句的替代實現方式分析

    本篇文章是對goto語句的替代實現方式進行了詳細的分析介紹,需要的朋友參考下...

    C語言教程網7342020-12-03
  • C/C++C語言實現電腦關機程序

    C語言實現電腦關機程序

    這篇文章主要為大家詳細介紹了C語言實現電腦關機程序,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下...

    xiaocaidayong8482021-08-20
  • C/C++詳解c語言中的 strcpy和strncpy字符串函數使用

    詳解c語言中的 strcpy和strncpy字符串函數使用

    strcpy 和strcnpy函數是字符串復制函數。接下來通過本文給大家介紹c語言中的strcpy和strncpy字符串函數使用,感興趣的朋友跟隨小編要求看看吧...

    spring-go5642021-07-02
  • C/C++c++ 單線程實現同時監聽多個端口

    c++ 單線程實現同時監聽多個端口

    這篇文章主要介紹了c++ 單線程實現同時監聽多個端口的方法,幫助大家更好的理解和學習使用c++,感興趣的朋友可以了解下...

    源之緣11542021-10-27
  • C/C++C/C++經典實例之模擬計算器示例代碼

    C/C++經典實例之模擬計算器示例代碼

    最近在看到的一個需求,本以為比較簡單,但花了不少時間,所以下面這篇文章主要給大家介紹了關于C/C++經典實例之模擬計算器的相關資料,文中通過示...

    jia150610152021-06-07
  • C/C++C語言中炫酷的文件操作實例詳解

    C語言中炫酷的文件操作實例詳解

    內存中的數據都是暫時的,當程序結束時,它們都將丟失,為了永久性的保存大量的數據,C語言提供了對文件的操作,這篇文章主要給大家介紹了關于C語言中文件...

    針眼_6702022-01-24
  • C/C++C++之重載 重定義與重寫用法詳解

    C++之重載 重定義與重寫用法詳解

    這篇文章主要介紹了C++之重載 重定義與重寫用法詳解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下...

    青山的青6062022-01-04
Weibo Article 1 Weibo Article 2 Weibo Article 3 Weibo Article 4 Weibo Article 5 Weibo Article 6 Weibo Article 7 Weibo Article 8 Weibo Article 9 Weibo Article 10 Weibo Article 11 Weibo Article 12 Weibo Article 13 Weibo Article 14 Weibo Article 15 Weibo Article 16 Weibo Article 17 Weibo Article 18 Weibo Article 19 Weibo Article 20 Weibo Article 21 Weibo Article 22 Weibo Article 23 Weibo Article 24 Weibo Article 25 Weibo Article 26 Weibo Article 27 Weibo Article 28 Weibo Article 29 Weibo Article 30 Weibo Article 31 Weibo Article 32 Weibo Article 33 Weibo Article 34 Weibo Article 35 Weibo Article 36 Weibo Article 37 Weibo Article 38 Weibo Article 39 Weibo Article 40
主站蜘蛛池模板: 久久久精品网站 | 日韩免费看 | 精品久久99| 久久精品日 | 红杏首页 | 亚洲精品一二区 | 中文字幕在线免费看 | 国产一区二区三区视频在线观看 | 99re6在线视频精品免费 | 日日操夜夜操天天操 | 综合五月网 | 亚洲91av | 久久久久一区 | 成人精品电影 | 日韩欧美自拍 | 欧美日韩第一页 | 精品国产一区二区三区日日嗨 | 国产青青草 | 欧美日韩免费在线 | 日韩欧美精品一区二区三区 | 国产激情午夜 | 欧美日韩在线播放 | 中文在线一区 | 国产精品成人3p一区二区三区 | 国产成人一区二区在线观看 | 91免费影视| 香蕉久久一区二区不卡无毒影院 | 天堂资源在线 | 日韩av一区在线 | 在线播放高清视频www | 偷偷干夜夜拍 | 国产亚洲精品久久久闺蜜 | 国产91精品一区二区绿帽 | 亚洲电影二区 | 精品视频一区在线观看 | 欧美视频一区二区三区 | 精品久久久av | 在线天堂av| 欧美一区二区三区视频在线 | 国产区视频 | eeuss国产一区二区三区四区 |