【字典樹】【KMP】【C++算法】3045統計前后綴下標對 II

作者推薦

動態規劃的時間復雜度優化

本文涉及知識點

字符串 字典樹 KMP 前后綴

LeetCode:3045統計前后綴下標對 II

給你一個下標從 0 開始的字符串數組 words 。
定義一個 布爾 函數 isPrefixAndSuffix ,它接受兩個字符串參數 str1 和 str2 :
當 str1 同時是 str2 的前綴(prefix)和后綴(suffix)時,isPrefixAndSuffix(str1, str2) 返回 true,否則返回 false。
例如,isPrefixAndSuffix(“aba”, “ababa”) 返回 true,因為 “aba” 既是 “ababa” 的前綴,也是 “ababa” 的后綴,但是 isPrefixAndSuffix(“abc”, “abcd”) 返回 false。
以整數形式,返回滿足 i < j 且 isPrefixAndSuffix(words[i], words[j]) 為 true 的下標對 (i, j) 的 數量 。
示例 1:
輸入:words = [“a”,“aba”,“ababa”,“aa”]
輸出:4
解釋:在本示例中,計數的下標對包括:
i = 0 且 j = 1 ,因為 isPrefixAndSuffix(“a”, “aba”) 為 true 。
i = 0 且 j = 2 ,因為 isPrefixAndSuffix(“a”, “ababa”) 為 true 。
i = 0 且 j = 3 ,因為 isPrefixAndSuffix(“a”, “aa”) 為 true 。
i = 1 且 j = 2 ,因為 isPrefixAndSuffix(“aba”, “ababa”) 為 true 。
因此,答案是 4 。
示例 2:

輸入:words = [“pa”,“papa”,“ma”,“mama”]
輸出:2
解釋:在本示例中,計數的下標對包括:
i = 0 且 j = 1 ,因為 isPrefixAndSuffix(“pa”, “papa”) 為 true 。
i = 2 且 j = 3 ,因為 isPrefixAndSuffix(“ma”, “mama”) 為 true 。
因此,答案是 2 。
示例 3:

輸入:words = [“abab”,“ab”]
輸出:0
解釋:在本示例中,唯一有效的下標對是 i = 0 且 j = 1 ,但是 isPrefixAndSuffix(“abab”, “ab”) 為 false 。
因此,答案是 0 。
提示:
1 <= words.length <= 105
1 <= words[i].length <= 105
words[i] 僅由小寫英文字母組成。
所有 words[i] 的長度之和不超過 5 * 105

分析

利用KMP 計算那些前綴等于后綴,然后在字典樹中查詢,此前綴(后綴)是否存在,如果存在根據編號查詢出現數量。
注意:前綴不能為空,可以等于本串。

代碼

核心代碼

template<class TData = char, int iTypeNum = 26, TData cBegin = 'a'>
class CTrieNode
{
public:CTrieNode* AddChar(TData ele, int& iMaxID){
#ifdef _DEBUGif ((ele < cBegin) || (ele >= cBegin + iTypeNum)){return nullptr;}
#endifconst int index = ele - cBegin;auto ptr = m_vPChilds[ele - cBegin];if (!ptr){m_vPChilds[index] = new CTrieNode();
#ifdef _DEBUGm_vPChilds[index]->m_iID = ++iMaxID;m_childForDebug[ele] = m_vPChilds[index];
#endif}return m_vPChilds[index];}CTrieNode* GetChild(TData ele)const{
#ifdef _DEBUGif ((ele < cBegin) || (ele >= cBegin + iTypeNum)){return nullptr;}
#endifreturn m_vPChilds[ele - cBegin];}
protected:
#ifdef _DEBUGint m_iID = -1;std::unordered_map<TData, CTrieNode*> m_childForDebug;
#endif
public:int m_iLeafIndex = -1;
protected:CTrieNode* m_vPChilds[iTypeNum] = { nullptr };
};template<class TData = char, int iTypeNum = 26, TData cBegin = 'a'>
class CTrie
{
public:int GetLeadCount(){return m_iLeafCount;}template<class IT>int Add(IT begin, IT end){auto pNode = &m_root;for (; begin != end; ++begin){pNode = pNode->AddChar(*begin, m_iMaxID);}if (-1 == pNode->m_iLeafIndex){pNode->m_iLeafIndex = m_iLeafCount++;}return pNode->m_iLeafIndex;}template<class IT>CTrieNode<TData, iTypeNum, cBegin>* Search(IT begin, IT end){auto ptr = &m_root;for (; begin != end; ++begin){ptr = ptr->GetChild(begin);if (nullptr == ptr){return nullptr;}}return ptr;}CTrieNode<TData, iTypeNum, cBegin> m_root;
protected:int m_iMaxID = 0;int m_iLeafCount = 0;
};class KMP
{
public:virtual int Find(const string& s, const string& t){CalLen(t);m_vSameLen.assign(s.length(), 0);for (int i1 = 0, j = 0; i1 < s.length(); ){for (; (j < t.length()) && (i1 + j < s.length()) && (s[i1 + j] == t[j]); j++);//i2 = i1 + j 此時s[i1,i2)和t[0,j)相等 s[i2]和t[j]不存在或相等m_vSameLen[i1] = j;//t[0,j)的結尾索引是j-1,所以最長公共前綴為m_vLen[j-1],簡寫為y 則t[0,y)等于t[j-y,j)等于s[i2-y,i2)if (0 == j){i1++;continue;}const int i2 = i1 + j;j = m_vLen[j - 1];i1 = i2 - j;//i2不變}for (int i = 0; i < m_vSameLen.size(); i++){//多余代碼是為了增加可測試性if (t.length() == m_vSameLen[i]){return i;}}return -1;}vector<int> m_vSameLen;//m_vSame[i]記錄 s[i...]和t[0...]最長公共前綴,增加可調試性static vector<int> Next(const string& s){const int len = s.length();vector<int> vNext(len, -1);for (int i = 1; i < len; i++){int next = vNext[i - 1];while ((-1 != next) && (s[next + 1] != s[i])){next = vNext[next];}vNext[i] = next + (s[next + 1] == s[i]);}return vNext;}
protected:void CalLen(const string& str){m_vLen.resize(str.length());for (int i = 1; i < str.length(); i++){int next = m_vLen[i - 1];while (str[next] != str[i]){if (0 == next){break;}next = m_vLen[next-1];}m_vLen[i] = next + (str[next] == str[i]);}}int m_c;vector<int> m_vLen;//m_vLen[i] 表示t[0,i]的最長公共前后綴	
};class Solution {
public:long long countPrefixSuffixPairs(vector<string>& words) {CTrie<> trie;unordered_map<int, int> mNoNum;long long llRet = 0;for (const auto& str : words){			KMP kmp;kmp.Find(str, str);queue<int> indexs;for (int i = str.length()-1; i >= 0 ; i--){if (kmp.m_vSameLen[i] == (str.length() - i)){indexs.emplace(str.length() - i);}}auto ptr = &trie.m_root;for (int i = 0; i < str.length(); i++){ptr = ptr->GetChild(str[i]);if (nullptr == ptr){break;}if ((-1 != ptr->m_iLeafIndex)&&indexs.size()&&( indexs.front()==i+1 )){llRet += mNoNum[ptr->m_iLeafIndex];					}while (indexs.size() && (indexs.front() == i + 1)){indexs.pop();}}mNoNum[trie.Add(str.begin(), str.end())]++;}return llRet;}
};

擴展閱讀

視頻課程

有效學習:明確的目標 及時的反饋 拉伸區(難度合適),可以先學簡單的課程,請移步CSDN學院,聽白銀講師(也就是鄙人)的講解。
https://edu.csdn.net/course/detail/38771

如何你想快速形成戰斗了,為老板分憂,請學習C#入職培訓、C++入職培訓等課程
https://edu.csdn.net/lecturer/6176

相關下載

想高屋建瓴的學習算法,請下載《喜缺全書算法冊》doc版
https://download.csdn.net/download/he_zhidan/88348653

我想對大家說的話
聞缺陷則喜是一個美好的愿望,早發現問題,早修改問題,給老板節約錢。
子墨子言之:事無終始,無務多業。也就是我們常說的專業的人做專業的事。
如果程序是一條龍,那算法就是他的是睛

測試環境

操作系統:win7 開發環境: VS2019 C++17
或者 操作系統:win10 開發環境: VS2022 C++17
如無特殊說明,本算法用**C++**實現。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/710947.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/710947.shtml
英文地址,請注明出處:http://en.pswp.cn/news/710947.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

C++——內存管理(new和delete)詳解

目錄 C/C內存管理 案例&#xff1a;變量在內存中到底會在哪&#xff1f; New和delete Operator new和operator delete函數 New和delete的原理 對內置類型 對自定義類型 定位new New/delete和malloc/free的區別 C/C內存管理 C/C內存管理分布圖&#xff1a;&#xff08;從…

項目案例:圖像分類技術在直播電商中的應用與實踐

一、引言 在數字化浪潮的推動下&#xff0c;電商行業迎來了一場革命性的變革。直播電商&#xff0c;作為一種新興的購物模式&#xff0c;正以其獨特的互動性和娛樂性&#xff0c;重塑著消費者的購物習慣。通過實時的直播展示&#xff0c;商品的細節得以清晰呈現&#xff0c;而互…

matlab:涉及復雜函數圖像的交點求解

matlab&#xff1a;涉及復雜函數圖像的交點求解 在MATLAB中求解兩個圖像的交點是一個常見的需求。本文將通過一個示例&#xff0c;展示如何求解兩個圖像的交點&#xff0c;并提供相應的MATLAB代碼。 畫出圖像 首先&#xff0c;我們需要繪制兩個圖像&#xff0c;以便直觀地看…

【JavaEE】_HttpServletResponse類

目錄 1. 核心方法 2. 關于setStatus(400)與sendError 2.1 setStatus(400) 2.2 sendError 3. setHeader方法 4. 構造重定向響應 4.1 使用setHeader和setStatus實現重定向 4.2 使用sendRedirect實現重定向 本專欄已有文章介紹HttpServlet和HttpServletRequest類&#…

仿真科普|CAE技術賦能無人機 低空經濟蓄勢起飛

喝一杯無人機送來的現磨熱咖啡&#xff1b;在擁堵的早高峰打個“空中的士”上班&#xff1b;乘坐水陸兩棲飛機來一場“陸海空”立體式觀光……曾經只出現在科幻片里的5D城市魔幻場景&#xff0c;正逐漸走進現實。而推動上述場景實現的&#xff0c;就是近年來越來越熱的“低空經…

前端開發——ElementUI組件的使用

文章目錄 1. Tabs標簽頁2. 單選框 el-radio3. 復選框 el-checkbox4. 下拉框 el-select5. 表格 el-table6. 對話框 el-dialog7. 文字提示 el-tooltip8. 抽屜 el-drawer 1. Tabs標簽頁 <template><el-tabs v-model"activeName" tab-click"handleClick&q…

python學生成績管理系統(期末課程作業)

功能介紹 平臺采用B/S結構&#xff0c;后端采用主流的Python語言進行開發&#xff0c;前端采用主流的Vue.js進行開發。本學期的期末作業。開發了1周 功能包括&#xff1a;成績管理、學生管理、課程管理、班級管理、用戶管理、日志管理、系統信息模塊。 源碼地址 https://gi…

c語言求簡單交錯序列前N項和

本題要求編寫程序,計算序列 1 - 1/4 1/7 - 1/10 ... 的前N項之和。 輸入格式: 輸入在一行中給出一個正整數N。 輸出格式: 在一行中按照“sum S”的格式輸出部分和的值S&#xff0c;精確到小數點后三位。題目保證計算結果不超過雙精度范圍。 輸入樣例: 10輸出樣例: su…

如何實現WordPress后臺顯示文章、分類目錄、標簽等的ID?

我們平時在使用WordPress的過程中&#xff0c;偶爾需要用到文章的ID&#xff0c;或分類目錄ID&#xff0c;或標簽ID&#xff0c;或媒體庫ID&#xff0c;或評論ID&#xff0c;或用戶ID等&#xff0c;但是WordPress后臺默認是不顯示它們的ID的。 今天boke112百科就跟大家分享如何…

聚觀早報 | 愛奇藝2023年Q4財報;蘋果將加大AI投入

聚觀早報每日整理最值得關注的行業重點事件&#xff0c;幫助大家及時了解最新行業動態&#xff0c;每日讀報&#xff0c;就讀聚觀365資訊簡報。 整理丨Cutie 3月1日消息 愛奇藝2023年Q4財報 蘋果將加大AI投入 意大利正與多家車企談判 多家企業與百度達成合作 比亞迪宋PL…

Cesium 視頻貼圖

一、創作靈感 a、在cesium中視頻或者圖像在矩形或者圓形中顯示 b、在不使用entity模式下,使用Primitive進行視頻或者圖像渲染 c、在使用Primitive的前提下,需要進行視頻或者圖像貼地 d、不貼地,請跳轉到我的另外一份日志紋理貼圖 二、創建步驟 1、創建圓形或者矩形 創建圓…

SpringBoot集成接口重試Retry

SpringBoot集成接口重試Retry 前言 在實際的應用中&#xff0c;我們經常需要調用第三方API來獲取數據或執行某些操作。然而&#xff0c;由于網絡不穩定、第三方服務異常等原因&#xff0c;API調用可能會失敗。為了提高系統的穩定性和可靠性&#xff0c;我們通常會考慮實現重試…

SDR架構 (一)為什么基帶有I和Q路?

我之前做過自己的RTL-SDR。一直有一個疑惑。為啥rtl2832u芯片有一對差分I路&#xff0c;還有一對差分Q路。差分很好理解是為了抗干擾&#xff0c;但為啥要I和Q呢&#xff1f;并且我也知道不少人在自己修改的時候&#xff0c;保留I路對接在r820t2&#xff08;跟原版一樣&#xf…

整數與IP地址間的轉換(牛客網算法/Javascript Node)

描述 原理&#xff1a;ip地址的每段可以看成是一個0-255的整數&#xff0c;把每段拆分成一個二進制形式組合起來&#xff0c;然后把這個二進制數轉變成 一個長整數。 舉例&#xff1a;一個ip地址為10.0.3.193 每段數字 相對應的二進制數 10 00001010 0 00000000 3 00000011 193…

開放簽電子簽章企業版上線【移動端功能(v1.5版本)】

春節序曲奏響創新華章&#xff0c;緊鑼密鼓的工作節奏下&#xff0c;開放簽支持移動端簽署啦&#xff01; 在這個萬家燈火的春節之際&#xff0c;開放簽團隊憑借高效的團隊協作&#xff0c;在節日的熱烈氛圍中成功推出了全新版本&#xff08;企業版1.5版&#xff09;&#xff…

逆變器專題(12)-弱電網

相應仿真原件請移步資源下載 通常情況下&#xff0c;理想電網都為強電網&#xff0c;但隨著光伏并網系統的大力發展&#xff0c;分布式光伏也越發鼎盛&#xff0c;越來越多的電力電子設備接入大電網、并且考慮能源利用問題&#xff0c;大部分光伏電站都建在戈壁沙漠等地區&…

多行業萬能預約門店小程序源碼系統 支持多門店預約小程序 帶完整的安裝代碼包以及搭建教程

隨著消費者對于服務體驗要求的不斷提升&#xff0c;門店預約系統成為了許多行業提升服務質量、提高運營效率的重要工具。然而&#xff0c;市面上的預約系統往往功能單一&#xff0c;無法滿足多行業、多場景的個性化需求。下面&#xff0c;小編集合了多年的行業經驗和技術積累&a…

巖土工程中的振弦采集儀技術發展與前景展望

巖土工程中的振弦采集儀技術發展與前景展望 河北穩控科技振弦采集儀是一種常用的巖土工程監測儀器&#xff0c;用于測量土壤或巖石的振動特性。隨著巖土工程領域的發展和技術的進步&#xff0c;振弦采集儀技術也得到了不斷的發展和改進。以下是對振弦采集儀技術發展與前景的展…

css5定位

css 一.定位1.概念&#xff08;定位定位模式邊位移&#xff09;2.靜態位移static&#xff08;不常用&#xff09;3.相對定位relative&#xff08;不脫標&#xff09;&#xff08;占位置&#xff09;4.絕對定位absolute&#xff08;脫標&#xff09;&#xff08;不占位置&#x…

VScode 單步斷點調試Nodejs方法總結

目錄 方法一 方法二 方法三 方法一 使用vscode開發nodejs程序,能夠啟動單步調試模式,在指定代碼處添加斷點,像chrome、firefox瀏覽器上一樣進行JavaScript的調試。 新建一個nodejs的工程,編寫代碼后,配置代碼調試的步驟: 1、切換到代碼調試界面 2、界面提示,新建一…