C++實(shí)現(xiàn)LeetCode(76.最小窗口子串)
[LeetCode] 76. Minimum Window Substring 最小窗口子串
Given a string S and a string T, find the minimum window in S which will contain all the characters in T in complexity O(n).
Example:
Input: S = "ADOBECODEBANC", T = "ABC"
Output: "BANC"
Note:
- If there is no such window in S that covers all characters in T, return the empty string
""
. - If there is such window, you are guaranteed that there will always be only one unique minimum window in S.
這道題給了我們一個(gè)原字符串S,還有一個(gè)目標(biāo)字符串T,讓在S中找到一個(gè)最短的子串,使得其包含了T中的所有的字母,并且限制了時(shí)間復(fù)雜度為 O(n)。這道題的要求是要在 O(n) 的時(shí)間度里實(shí)現(xiàn)找到這個(gè)最小窗口字串,暴力搜索 Brute Force 肯定是不能用的,因?yàn)楸闅v所有的子串的時(shí)間復(fù)雜度是平方級(jí)的。那么來(lái)想一下,時(shí)間復(fù)雜度卡的這么嚴(yán),說(shuō)明必須在一次遍歷中完成任務(wù),當(dāng)然遍歷若干次也是 O(n),但不一定有這個(gè)必要,嘗試就一次遍歷拿下!那么再來(lái)想,既然要包含T中所有的字母,那么對(duì)于T中的每個(gè)字母,肯定要快速查找是否在子串中,既然總時(shí)間都卡在了 O(n),肯定不想在這里還浪費(fèi)時(shí)間,就用空間換時(shí)間(也就算法題中可以這么干了,七老八十的富翁就算用大別野也換不來(lái)時(shí)間啊。依依東望,望的就是時(shí)間吶 T.T),使用 HashMap,建立T中每個(gè)字母與其出現(xiàn)次數(shù)之間的映射,那么你可能會(huì)有疑問(wèn),為啥不用 HashSet 呢,別急,講到后面你就知道用 HashMap 有多妙,簡(jiǎn)直妙不可言~
目前在腦子一片漿糊的情況下,我們還是從簡(jiǎn)單的例子來(lái)分析吧,題目例子中的S有點(diǎn)長(zhǎng),換個(gè)短的 S = "ADBANC",T = "ABC",那么肉眼遍歷一遍S唄,首先第一個(gè)是A,嗯很好,T中有,第二個(gè)是D,T中沒(méi)有,不理它,第三個(gè)是B,嗯很好,T中有,第四個(gè)又是A,多了一個(gè),禮多人不怪嘛,收下啦,第五個(gè)是N,一邊涼快去,第六個(gè)終于是C了,那么貌似好像需要整個(gè)S串,其實(shí)不然,注意之前有多一個(gè)A,就算去掉第一個(gè)A,也沒(méi)事,因?yàn)榈谒膫€(gè)A可以代替之,第二個(gè)D也可以去掉,因?yàn)椴辉赥串中,第三個(gè)B就不能再去掉了,不然就沒(méi)有B了。所以最終的答案就"BANC"了。通過(guò)上面的描述,你有沒(méi)有發(fā)現(xiàn)一個(gè)有趣的現(xiàn)象,先擴(kuò)展,再收縮,就好像一個(gè)窗口一樣,先擴(kuò)大右邊界,然后再收縮左邊界,上面的例子中右邊界無(wú)法擴(kuò)大了后才開(kāi)始收縮左邊界,實(shí)際上對(duì)于復(fù)雜的例子,有可能是擴(kuò)大右邊界,然后縮小一下左邊界,然后再擴(kuò)大右邊界等等。這就很像一個(gè)不?;瑒?dòng)的窗口了,這就是大名鼎鼎的滑動(dòng)窗口 Sliding Window 了,簡(jiǎn)直是神器啊,能解很多子串,子數(shù)組,子序列等等的問(wèn)題,是必須要熟練掌握的?。?/p>
下面來(lái)考慮用代碼來(lái)實(shí)現(xiàn),先來(lái)回答一下前面埋下的伏筆,為啥要用 HashMap,而不是 HashSet,現(xiàn)在應(yīng)該很顯而易見(jiàn)了吧,因?yàn)橐y(tǒng)計(jì)T串中字母的個(gè)數(shù),而不是僅僅看某個(gè)字母是否在T串中出現(xiàn)。統(tǒng)計(jì)好T串中字母的個(gè)數(shù)了之后,開(kāi)始遍歷S串,對(duì)于S中的每個(gè)遍歷到的字母,都在 HashMap 中的映射值減1,如果減1后的映射值仍大于等于0,說(shuō)明當(dāng)前遍歷到的字母是T串中的字母,使用一個(gè)計(jì)數(shù)器 cnt,使其自增1。當(dāng) cnt 和T串字母?jìng)€(gè)數(shù)相等時(shí),說(shuō)明此時(shí)的窗口已經(jīng)包含了T串中的所有字母,此時(shí)更新一個(gè) minLen 和結(jié)果 res,這里的 minLen 是一個(gè)全局變量,用來(lái)記錄出現(xiàn)過(guò)的包含T串所有字母的最短的子串的長(zhǎng)度,結(jié)果 res 就是這個(gè)最短的子串。然后開(kāi)始收縮左邊界,由于遍歷的時(shí)候,對(duì)映射值減了1,所以此時(shí)去除字母的時(shí)候,就要把減去的1加回來(lái),此時(shí)如果加1后的值大于0了,說(shuō)明此時(shí)少了一個(gè)T中的字母,那么 cnt 值就要減1了,然后移動(dòng)左邊界 left。你可能會(huì)疑問(wèn),對(duì)于不在T串中的字母的映射值也這么加呀減呀的,真的大丈夫(帶膠布)嗎?其實(shí)沒(méi)啥事,因?yàn)閷?duì)于不在T串中的字母,減1后,變-1,cnt 不會(huì)增加,之后收縮左邊界的時(shí)候,映射值加1后為0,cnt 也不會(huì)減少,所以并沒(méi)有什么影響啦,下面是具體的步驟啦:
- 先掃描一遍T(mén),把對(duì)應(yīng)的字符及其出現(xiàn)的次數(shù)存到 HashMap 中。
- 然后開(kāi)始遍歷S,就把遍歷到的字母對(duì)應(yīng)的 HashMap 中的 value 減一,如果減1后仍大于等于0,cnt 自增1。
- 如果 cnt 等于T串長(zhǎng)度時(shí),開(kāi)始循環(huán),紀(jì)錄一個(gè)字串并更新最小字串值。然后將子窗口的左邊界向右移,如果某個(gè)移除掉的字母是T串中不可缺少的字母,那么 cnt 自減1,表示此時(shí)T串并沒(méi)有完全匹配。
解法一:
class Solution { public: string minWindow(string s, string t) { string res = ""; unordered_map<char, int> letterCnt; int left = 0, cnt = 0, minLen = INT_MAX; for (char c : t) ++letterCnt[c]; for (int i = 0; i < s.size(); ++i) { if (--letterCnt[s[i]] >= 0) ++cnt; while (cnt == t.size()) { if (minLen > i - left + 1) { minLen = i - left + 1; res = s.substr(left, minLen); } if (++letterCnt[s[left]] > 0) --cnt; ++left; } } return res; } };
這道題也可以不用 HashMap,直接用個(gè) int 的數(shù)組來(lái)代替,因?yàn)?ASCII 只有256個(gè)字符,所以用個(gè)大小為 256 的 int 數(shù)組即可代替 HashMap,但由于一般輸入字母串的字符只有 128 個(gè),所以也可以只用 128,其余部分的思路完全相同,雖然只改了一個(gè)數(shù)據(jù)結(jié)構(gòu),但是運(yùn)行速度提高了一倍,說(shuō)明數(shù)組還是比 HashMap 快啊。還可以進(jìn)一步的優(yōu)化,沒(méi)有必要每次都計(jì)算子串,只要有了起始位置和長(zhǎng)度,就能唯一的確定一個(gè)子串。這里使用一個(gè)全局變量 minLeft 來(lái)記錄最終結(jié)果子串的起始位置,初始化為 -1,最終配合上 minLen,就可以得到最終結(jié)果了。注意在返回的時(shí)候要檢測(cè)一下若 minLeft 仍為初始值 -1,需返回空串,參見(jiàn)代碼如下:
解法二:
class Solution { public: string minWindow(string s, string t) { vector<int> letterCnt(128, 0); int left = 0, cnt = 0, minLeft = -1, minLen = INT_MAX; for (char c : t) ++letterCnt[c]; for (int i = 0; i < s.size(); ++i) { if (--letterCnt[s[i]] >= 0) ++cnt; while (cnt == t.size()) { if (minLen > i - left + 1) { minLen = i - left + 1; minLeft = left; } if (++letterCnt[s[left]] > 0) --cnt; ++left; } } return minLeft == -1 ? "" : s.substr(minLeft, minLen); } };
到此這篇關(guān)于C++實(shí)現(xiàn)LeetCode(76.最小窗口子串)的文章就介紹到這了,更多相關(guān)C++實(shí)現(xiàn)最小窗口子串內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C++實(shí)現(xiàn)LeetCode(85.最大矩形)
- C++實(shí)現(xiàn)LeetCode(82.移除有序鏈表中的重復(fù)項(xiàng)之二)
- C++實(shí)現(xiàn)LeetCode(81.在旋轉(zhuǎn)有序數(shù)組中搜索之二)
- C++實(shí)現(xiàn)LeetCode(80.有序數(shù)組中去除重復(fù)項(xiàng)之二)
- C++實(shí)現(xiàn)LeetCode(79.詞語(yǔ)搜索)
- C++實(shí)現(xiàn)LeetCode(75.顏色排序)
- C++實(shí)現(xiàn)LeetCode(74.搜索一個(gè)二維矩陣)
- C++實(shí)現(xiàn)LeetCode(73.矩陣賦零)
- C++實(shí)現(xiàn)LeetCode(137.單獨(dú)的數(shù)字之二)
相關(guān)文章
C++ 詳細(xì)講解stack與queue的模擬實(shí)現(xiàn)
C++ Stack(堆棧) 是一個(gè)容器類(lèi)的改編,為程序員提供了堆棧的全部功能,也就是說(shuō)實(shí)現(xiàn)了一個(gè)先進(jìn)后出(FILO)的數(shù)據(jù)結(jié)構(gòu),許多程序都使用了 queue 容器。queue 容器可以用來(lái)表示超市的結(jié)賬隊(duì)列或服務(wù)器上等待執(zhí)行的數(shù)據(jù)庫(kù)事務(wù)隊(duì)列2022-04-04Qt中QStringList與QString的常用方法總結(jié)
這篇文章主要為大家總結(jié)了Qt中QString 與 (QStringList | QByteArray)之間的轉(zhuǎn)換,以及QString、QStringList的一些常用方法,感興趣的可以收藏一下2022-12-12數(shù)據(jù)結(jié)構(gòu) 中數(shù)制轉(zhuǎn)換(棧的應(yīng)用)
這篇文章主要介紹了數(shù)據(jù)結(jié)構(gòu) 中數(shù)制轉(zhuǎn)換(棧的應(yīng)用)的相關(guān)資料,需要的朋友可以參考下2017-06-06Qt數(shù)據(jù)庫(kù)應(yīng)用之實(shí)現(xiàn)通用數(shù)據(jù)庫(kù)清理
項(xiàng)目如果需要存儲(chǔ)很多日志記錄比如運(yùn)行日志,時(shí)間長(zhǎng)了記錄數(shù)量非常多,數(shù)據(jù)庫(kù)體積不斷增大,對(duì)應(yīng)數(shù)據(jù)庫(kù)表的增刪改查的效率不斷降低,因此需要將早期的數(shù)據(jù)清理。本文將詳細(xì)介紹一下通用數(shù)據(jù)庫(kù)清理的實(shí)現(xiàn),需要的可以參考一下2022-02-02C語(yǔ)言細(xì)致講解線(xiàn)程同步的集中方式
多線(xiàn)程中的線(xiàn)程同步可以使用,CreateThread,CreateMutex 互斥鎖實(shí)現(xiàn)線(xiàn)程同步,通過(guò)臨界區(qū)實(shí)現(xiàn)線(xiàn)程同步,Semaphore 基于信號(hào)實(shí)現(xiàn)線(xiàn)程同步,CreateEvent 事件對(duì)象的同步,以及線(xiàn)程函數(shù)傳遞單一參數(shù)與多個(gè)參數(shù)的實(shí)現(xiàn)方式2022-05-05QT實(shí)戰(zhàn)之打開(kāi)最近圖片功能的實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了如何利用Qt和QSettings實(shí)現(xiàn)打開(kāi)最近圖片功能,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)QT有一定的幫助,感興趣的可以了解一下2022-06-06