摘要:例如,,則的其中一種組合是的子串,然后返回。如果從題目給出的例子來窮舉,一共種組合,很容易窮舉出來,但是字符串長度非常大的時候,怎么辦呢所以,窮舉的辦法被我排除了。
題目要求
這道算法題在前端面試中可能遇到,據(jù)說某條出過這題。
查找字符串B的字符任意一種組合是否是字符串A的子串。算法思路
例如 A=abc123,B=cba,則B的其中一種組合abc是A的子串,然后返回true。
題目的出處已經無從考究,接下來我們從JavaScript的角度來封裝這樣一個功能函數(shù)。
窮舉一開始看到這道題,你會想到什么?
我想到的是先列舉出B的所有排列組合,存到數(shù)組里面,然后遍歷,判斷是否有組合包含在A中,如果有返回true,否則返回false。
如果從題目給出的例子來窮舉,一共6種組合,很容易窮舉出來,但是字符串長度非常大的時候,怎么辦呢?
所以,窮舉的辦法被我排除了。
這名字聽起來很奇怪,怎么個思路呢?
1、A的排序肯定是不變的,既然可變的我們很難下手,那么可以從不變的地方入手,以不變應萬變。
2、看字符串可能不太習慣,我把A和B都轉換成數(shù)組。
let a = A.split("") // [a, b, c, 1, 2, 3] let b = B.split("") // [c, b, a]
3、先過濾數(shù)組為空的情況,如果a或者b為空,那么不需要比較,返回false。
if (a.length === 0 || b.length === 0) { return false }
4、只看數(shù)組b,可以有6種排列組合,[c,b,a],[a,b,c],[a,c,b],[b,a,c],[b,c,a],[c,a,b]。還記得第1步說的,我們不管b有多少種組合,都從a入手。
// a = [a, b, c, 1, 2, 3] for (let j = 0; j < a.length; j++) { }
5、遍歷a有什么作用呢?接下來我為大家揭曉何為標記刪除法,允許我小小解釋一下該方法,分為2個核心,“標記”和“刪除”,“標記”是指標記當前數(shù)組a遍歷的位置,“刪除”是指刪除數(shù)組b中的元素。
這樣說可能不太懂,先不看代碼,我用數(shù)組來模擬一下執(zhí)行過程。
初始化的值 a = [a, b, c, 1, 2, 3] b = [c, b, a] ================================================== 當遍歷a的時候,j從0開始,遍歷到a.length-1結束 ================================================== j = 0 // 給a里的字符加"",做個標記,表示當前遍歷的下標位置 a = ["a", b, c, 1, 2, 3] ================================================== 然后我們發(fā)現(xiàn)數(shù)組b存在當前的字符"a",執(zhí)行刪除操作 b = [c, b] ================================================== j = 1 // 數(shù)組a遍歷到第二個字符 a = [a, "b", c, 1, 2, 3] // 標記 b = [b] // 刪除 ================================================== j = 1 // 數(shù)組a遍歷到第三個字符 a = [a, b, "c", 1, 2, 3] // 標記 b = [] // 刪除 ================================================== 現(xiàn)在我們看到b數(shù)組變成空了,則證明b的任意一種排列存在于a中
6、上一步描述的情況是最簡單的狀態(tài),剛好在A字符中存在不間斷的字符組合。我們把A改一下,變成 A = a1b2c3abc。即使變復雜了,我們依舊可以使用標記刪除發(fā)來做,只是稍微做一點處理。
初始化的值 a = [a, 1, b, 2, c, 3, a, b, c] b = [c, b, a] ================================================== 當遍歷a的時候,j從0開始,遍歷到a.length-1結束 ================================================== j = 0 // 給a里的字符加"",做個標記,表示當前遍歷的下標位置 a = ["a", 1, b, 2, c, 3, a, b, c] ================================================== 然后我們發(fā)現(xiàn)數(shù)組b存在當前的字符"a",執(zhí)行刪除操作 b = [c, b] ================================================== j = 1 // 數(shù)組a遍歷到第二個字符 a = [a, "1", b, 2, c, 3, a, b, c] // 標記 // 突然發(fā)現(xiàn)第2個字符是1,現(xiàn)在該怎么辦?我們只需要把數(shù)組b恢復初始狀態(tài)即可 b = [c, b, a] // 恢復初始狀態(tài) ================================================== 接下來繼續(xù)遍歷,重復上面的處理步驟,直到數(shù)組b為空或者數(shù)組a遍歷完成,返回結果
7、JavaScript代碼實現(xiàn)
下面是第6步說明的代碼實現(xiàn),從代碼中可以看到,不管B字符有多少排列組合,我們始終只需要遍歷A的所有字符即可,內部實現(xiàn)也是用空間換時間。
// 遍歷數(shù)組 a for (let j = 0; j < a.length; j++) { // 數(shù)組 b不為空,下一步 if (b.length > 0) { // 數(shù)組a存在當前遍歷的數(shù)組b的元素 if (b.indexOf(a[j]) > -1) { // 刪除b數(shù)組中匹配到的第一個對應下標的元素 b.splice(b.indexOf(a[j]), 1) if (b.length === 0) { // 如果數(shù)組b全部被刪除了,則證明b是a的子串 return true } } else { // 數(shù)組b不存在當前遍歷的數(shù)組b的元素,恢復b數(shù)組 b = B.split("") } } else { // 數(shù)組 b為空返回true return true } }總結
與其他前端工程師的交流中,我還了解到了其他的解題思路,有些很有趣,比如考慮使用Map或Set、滑塊區(qū)間比較等,不過我沒有去用代碼實現(xiàn)過,如果你有其他的方法,可以在下面留言交流。
完整源碼評論區(qū)有人指出不能覆蓋某些場景的測試用例,所以我對上面的具體過程做了改進,下面是改進后的源碼。
增加了2個字段,isBack和isRestart,isRestart用來標記是否重新在當前位置遍歷,isBack判斷是否對數(shù)組遍歷的下標進行回退一個單位。
var A = "abc123" , B = "cba" function interface(A, B) { // 將A和B轉成數(shù)組 let a = A.split("") let b = B.split("") if (a.length === 0 || b.length === 0) { return false } let isBack = false, isRestart = 0 // 遍歷數(shù)組 a for (let j = 0; j < a.length; j++) { // 數(shù)組 b不為空,下一步 if (b.length > 0) { isBack = false // 數(shù)組a存在當前遍歷的數(shù)組b的元素 if (b.indexOf(a[j]) > -1) { // 刪除b數(shù)組中匹配到的第一個對應下標的元素 b.splice(b.indexOf(a[j]), 1) if (b.length === 0) { // 如果數(shù)組b全部被刪除了,則證明b是a的子串 return true } } else { if (isRestart !== 0) { isBack = false } else { isBack = true } // 數(shù)組b不存在當前遍歷的數(shù)組b的元素,恢復b數(shù)組 b = B.split("") if (isBack) { j -= 1 isRestart = 0 } isRestart++ } } else { // 數(shù)組 b為空返回true return true } } return false } interface(A, B)
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/52996.html
摘要:例如,,則的其中一種組合是的子串,然后返回。如果從題目給出的例子來窮舉,一共種組合,很容易窮舉出來,但是字符串長度非常大的時候,怎么辦呢所以,窮舉的辦法被我排除了。 題目要求 這道算法題在前端面試中可能遇到,據(jù)說某條出過這題。 查找字符串B的字符任意一種組合是否是字符串A的子串。例如 A=abc123,B=cba,則B的其中一種組合abc是A的子串,然后返回true。 算法思路 題目的...
摘要:正則表達式是由普通字符例如字符到以及特殊字符稱為元字符組成的文字模式。方法參數(shù)一個正則表達式對象。如果正則表達式沒有標志,則會返回和相同的結果。其被視為一整個字符串,而不是一個正則表達式。 正則表達式 正則表達式(Regular Expression)是一種文本模式,包括普通字符(例如,a 到 z 之間的字母)和特殊字符(稱為元字符)。正則表達式使用單個字符串來描述、匹配一系列匹配某個...
摘要:正則表達式就是用來描述他稱為正則集的代數(shù)的表達式,因此采用正則表達式這個術語。文本中正則表達式結束搜索的索引值與和方法的同名參數(shù)相同。對象是一個編號的正則表達式通過提供的一系列方法可以對文本進行匹配查找。 一、概述 今天這篇文章帶領大家學習一下Python中的正則表達式,當然了,正則表達式本身的內容就足以寫好幾本書了,我們這里列出的內容,僅僅是Python中常用的和基礎的一些內容。 那...
摘要:語法參數(shù)必填項,字符串或正則表達式,該參數(shù)指定的地方分割可選該參數(shù)指定返回的數(shù)組的最大長度,如果設置了該參數(shù),返回的子字符串不會多于這個參數(shù)指定的數(shù)組。該數(shù)組通過在指定的邊界處將字符串分割成子字符串。把正則表達式拆分成小表達式。 正則表達式是什么 RegExp 對象表示正則表達式,它是對字符串執(zhí)行模式匹配的強大工具。 為什么使用正則表達式 測試字符串內的模式。例如,可以測試輸入字符串...
摘要:引用就是允許在同一個正則表達式的后部引用前面的子表達式。這個數(shù)字制定了帶圓括號的子表達式在正則表達式中的位置。對正則表達式中前一個子表達式的引用,并不是指對子表達式模式的引用,而是指與那個模式匹配的文本的引用。 前言 本文主要是在讀《JavaScript高級程序語言設計》一書有關正則表達式的章節(jié)的知識點記錄,方便后續(xù)查閱。 什么是正則表達式 正則表達式是用來描述字符組合的某種規(guī)則。它可...
閱讀 3877·2021-07-28 18:10
閱讀 2585·2019-08-30 15:44
閱讀 1098·2019-08-30 14:07
閱讀 3468·2019-08-29 17:20
閱讀 1587·2019-08-26 18:35
閱讀 3543·2019-08-26 13:42
閱讀 1827·2019-08-26 11:58
閱讀 1601·2019-08-23 18:33