[LeetCode] 392. 判斷子序列 ☆(動態規划)


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;
    }

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM