https://leetcode-cn.com/problems/is-subsequence/solution/java-dp-by-zxy0917-5/
描述
給定字符串 s 和 t ,判斷 s 是否為 t 的子序列。
你可以認為 s 和 t 中僅包含英文小寫字母。字符串 t 可能會很長(長度 ~= 500,000),而 s 是個短字符串(長度 <=100)。
字符串的一個子序列是原始字符串刪除一些(也可以不刪除)字符而不改變剩余字符相對位置形成的新字符串。(例如,"ace"是"abcde"的一個子序列,而"aec"不是)。
示例 1:
s = "abc", t = "ahbgdc"
返回 true.
示例 2:
s = "axc", t = "ahbgdc"
返回 false.
后續挑戰 :
如果有大量輸入的 S,稱作S1, S2, ... , Sk 其中 k >= 10億,你需要依次檢查它們是否為 T 的子序列。在這種情況下,你會怎樣改變代碼?
解析
直接思路
直接遍歷t,一個指針。比較即可。
動態規划
子序列的解法,都可以使用動態規划。
生成一個二維數組 boolean[][] dp = new boolean[sLen + 1][tLen + 1]; s的從頭開始到i的子字符串是否為t從頭開始到j的子字符串的子序列
當s長度為0,肯定是t的子序列。即 dp[0][j] = true;
狀態轉移公式:
當char[i]==char[j]時,則字符i一定是j的子序列。如果此時0~i-1子字符串是0~j-1子字符串的子序列(即dp[i-1][j-1]=true),那么dp[i][j]=true。所以dp[i][j] = dp[i-1][j-1];
當char[i]!=char[j]時,即判斷當前0~i子字符串是否是0~j-1的子字符串的子序列,即dp[i][j] = dp[i][j - 1]。
如ab,eabc,雖然s的最后一個字符和t中最后一個字符不相等,但是因為ab是eab的子序列,所以ab也是eabc的子序列
代碼
直接思路
public boolean isSubsequence(String s, String t) { int sLen = s.length(); int tLen = t.length(); if (sLen <= 0) { return true; } else if (sLen > tLen) { return false; } int ss = 0; for (int i = 0; i < tLen; i++) { if (s.charAt(ss) == t.charAt(i)) { ss++; if (ss == sLen) { return true; } } } return ss == sLen;
動態規划
public boolean isSubsequence(String s, String t) { int sLen = s.length(); int tLen = t.length(); if (sLen <= 0) { return true; } else if (sLen > tLen) { return false; } boolean[][] dp = new boolean[sLen + 1][tLen + 1]; for (int i = 0; i < tLen; i++) { dp[0][i] = true; } for (int i = 1; i < sLen + 1; i++) { for (int j = 1; j < tLen + 1; j++) { if (s.charAt(i - 1) == t.charAt(j - 1)) { dp[i][j] = dp[i - 1][j - 1]; } else { dp[i][j] = dp[i][j - 1]; } } } return dp[sLen][tLen]; }
DP優化
由上面可見,當前的dp數據,只會受當前行,和前一行的影響,固可以用一維數組來處理。
public boolean isSubsequence(String s, String t) { if (s == null || s.length() <= 0) { return true; } if (t == null || s.length() > t.length()) { return false; } int sLen = s.length(); int tLen = t.length(); int[] dp = new int[tLen + 1]; int kStart = 1; for (int ii = 1; ii <= sLen; ii++) { boolean equalsFlag = false;//每次遍歷一行,最多進去1次 for (int kk = kStart; kk <= tLen; kk++) { if (!equalsFlag && s.charAt(ii - 1) == t.charAt(kk - 1)) { dp[kk] = dp[kk] + 1; kStart = kk + 1;// kk下次遍歷的開始處需大於當前位置 equalsFlag = true; } else { dp[kk] = dp[kk - 1]; } } } return dp[tLen] == sLen; }