亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

函數在某些情況下可以工作,但當最長的子字符串“重用”字符時會失敗

函數在某些情況下可以工作,但當最長的子字符串“重用”字符時會失敗

哆啦的時光機 2023-06-21 14:56:46
我有一個名為 lengthOfLongestSubstring 的函數,它的工作是找到沒有任何重復字符的最長子字符串。在大多數情況下,它可以工作,但是當它獲得像“dvdf”這樣的輸入時,它會打印出 2(而不是 3),并在應該是 [d, vdf] 時給出 [dv, df]。因此,我首先檢查該字符串,看看是否有任何獨特的字符。如果有,我將其附加到 ans 變量中。(我認為這是需要修復的部分)。如果存在重復項,我將其存儲在子字符串鏈表中,并將 ans 變量重置為重復字符串。遍歷整個字符串后,我找到最長的子字符串并返回它的長度。public static int lengthOfLongestSubstring(String s) {        String ans = "";    int len = 0;    LinkedList<String> substrings = new LinkedList<String>();     for (int i = 0; i < s.length(); i++) {      if (!ans.contains("" + s.charAt(i))) {        ans += s.charAt(i);      } else {        substrings.add(ans);        ans = "" + s.charAt(i);      }    }    substrings.add(ans); // add last seen substring into the linked list    for (int i = 0; i < substrings.size(); i++) {      if (substrings.get(i).length() >= len)        len = substrings.get(i).length();    }    System.out.println(Arrays.toString(substrings.toArray()));    return len;}下面是一些測試結果://correctlengthOfLongestSubstring("abcabcbb") -> 3 ( [abc, abc, b, b]) lengthOfLongestSubstring("pwwkew") -> 3 ([pw, wke, w]).lengthOfLongestSubstring("ABDEFGABEF"); -> 6 ([ABDEFG, ABEF]) // wrongSystem.out.println(lengthOfLongestSubstring("acadf")); -> 3, ([ac, adf]) *should be 4, with the linked list being [a, cadf]有什么建議來解決這個問題嗎?我必須重做所有邏輯嗎?
查看完整描述

3 回答

?
FFIVE

TA貢獻1797條經驗 獲得超6個贊

您的代碼錯誤地假設當您找到重復字符時,下一個候選子字符串從重復字符開始。這不是真的,它是在原始角色之后開始的。


示例:如果字符串為"abcXdefXghiXjkl",則有 3 個候選子串:"abcXdef"、"defXghi"、 和"ghiXjkl"。


正如您所看到的,候選子字符串在重復字符之前結束,并在重復字符之后開始(以及字符串的開頭和結尾)。


因此,當您找到重復字符時,需要該字符的前一個實例的位置來確定下一個候選子字符串的開頭。


處理這個問題的最簡單方法是將Map角色構建到上次看到的位置。這也比不斷執行子字符串搜索來檢查重復字符(就像問題代碼和其他答案所做的那樣)執行得更快。


像這樣的東西:


public static int lengthOfLongestSubstring(String s) {

    Map<Character, Integer> charPos = new HashMap<>();

    List<String> candidates = new ArrayList<>();

    int start = 0, maxLen = 0;

    for (int idx = 0; idx < s.length(); idx++) {

        char ch = s.charAt(idx);

        Integer preIdx = charPos.get(ch);

        if (preIdx != null && preIdx >= start) { // found repeat

            if (idx - start > maxLen) {

                candidates.clear();

                maxLen = idx - start;

            }

            if (idx - start == maxLen)

                candidates.add(s.substring(start, idx));

            start = preIdx + 1;

        }

        charPos.put(ch, idx);

    }

    if (s.length() - start > maxLen)

        maxLen = s.length() - start;

    if (s.length() - start == maxLen)

        candidates.add(s.substring(start));

    System.out.print(candidates + ": ");

    return maxLen;

}

僅candidates用于調試目的,并且不是必需的,因此如果沒有它,代碼會更簡單一些:


public static int lengthOfLongestSubstring(String s) {

    Map<Character, Integer> charPos = new HashMap<>();

    int start = 0, maxLen = 0;

    for (int idx = 0; idx < s.length(); idx++) {

        char ch = s.charAt(idx);

        Integer preIdx = charPos.get(ch);

        if (preIdx != null && preIdx >= start) { // found repeat

            if (idx - start > maxLen)

                maxLen = idx - start;

            start = preIdx + 1;

        }

        charPos.put(ch, idx);

    }

    return Math.max(maxLen, s.length() - start);

}

測試


System.out.println(lengthOfLongestSubstring(""));

System.out.println(lengthOfLongestSubstring("x"));

System.out.println(lengthOfLongestSubstring("xx"));

System.out.println(lengthOfLongestSubstring("xxx"));

System.out.println(lengthOfLongestSubstring("abcXdefXghiXjkl"));

System.out.println(lengthOfLongestSubstring("abcabcbb"));

System.out.println(lengthOfLongestSubstring("pwwkew"));

System.out.println(lengthOfLongestSubstring("ABDEFGABEF"));

輸出(帶有候選列表)


[]: 0

[x]: 1

[x, x]: 1

[x, x, x]: 1

[abcXdef, defXghi, ghiXjkl]: 7

[abc, bca, cab, abc]: 3

[wke, kew]: 3

[ABDEFG, BDEFGA, DEFGAB]: 6


查看完整回答
反對 回復 2023-06-21
?
滄海一幻覺

TA貢獻1824條經驗 獲得超5個贊

ans當找到字符匹配時,而不是設置為當前字符


ans = "" + s.charAt(i);

您應該將當前字符添加到當前字符的第一個匹配之后的所有字符


ans = ans.substring(ans.indexOf(s.charAt(i)) + 1) + s.charAt(i);

完整的方法因此變成


    public static int lengthOfLongestSubstring(String s) {

        String ans = "";

        int len = 0;

        LinkedList<String> substrings = new LinkedList<>();

        for (int i = 0; i < s.length(); i++) {

            if (!ans.contains("" + s.charAt(i))) {

                ans += s.charAt(i);

            } else {

                substrings.add(ans);

                // Only the below line changed

                ans = ans.substring(ans.indexOf(s.charAt(i)) + 1) + s.charAt(i);

            }

        }


        substrings.add(ans); // add last seen substring into the linked list


        for (int i = 0; i < substrings.size(); i++) {

            if (substrings.get(i).length() >= len)

                len = substrings.get(i).length();

        }


        System.out.println(Arrays.toString(substrings.toArray()));


        return len;

    }

使用此代碼,您指定的驗收標準已成功通過


//correct

lengthOfLongestSubstring("dvdf") -> 3 ( [dv, vdf]) 

lengthOfLongestSubstring("abcabcbb") -> 3 ([abc, bca, cab, abc, cb, b]) 

lengthOfLongestSubstring("pwwkew") -> 3 ([pw, wke, kew]).

lengthOfLongestSubstring("ABDEFGABEF"); -> 6 ([ABDEFG, BDEFGA, DEFGAB, FGABE, GABEF])

lengthOfLongestSubstring("acadf"); -> 4 ([ac, cadf])


查看完整回答
反對 回復 2023-06-21
?
陪伴而非守候

TA貢獻1757條經驗 獲得超8個贊

創建一個嵌套的 for 循環來檢查數組中的每個索引。


    public static int lengthOfLongestSubstring(String s) {    

    String ans = "";

    int len = 0;

    LinkedList<String> substrings = new LinkedList<String>();

    int k = 0;

    for (int i = 0; i < s.length(); i++) {

        if(k == s.length()) {

            break;

        }

        for(k = i; k < s.length(); k++) {

          if (!ans.contains("" + s.charAt(k))) {

            ans += s.charAt(k);

          } else {

            substrings.add(ans);

            ans = "";

            break;

          }

        }

    }


    substrings.add(ans); // add last seen substring into the linked list


    for (int i = 0; i < substrings.size(); i++) {

      if (substrings.get(i).length() >= len)

        len = substrings.get(i).length();

    }


    System.out.println(Arrays.toString(substrings.toArray()));


    return len;


}

例子:


lengthOfLongestSubstring("ABDEFGABEF"); -> 6 ([ABDEFG, BDEFGA, DEFGAB, EFGAB, FGABE, GABEF])



查看完整回答
反對 回復 2023-06-21
  • 3 回答
  • 0 關注
  • 197 瀏覽
慕課專欄
更多

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號