找出字符串中不含重復(fù)字符的最長子串的長度

1. 題目

給定一個字符串,找出其中不含重復(fù)字符的最長子串的長度。

示例1:
輸入:"abcabcbb"
輸出:3
解釋:因為無重復(fù)字符的最長子串是:"abc"谭跨,所以其長度為3

示例2:
輸入:"bbbbb"
輸出:1
解釋:因為無重復(fù)字符的最長子串是:"b",所以其長度為1

示例3:
輸入:"pwwkew"
輸出:3
解釋:因為無重復(fù)字符的最長子串是:"wke"或"kew",所以其長度為3
注意:必須是子串的長度囚灼,子串是連續(xù)的字符,中間不能跳躍字符祭衩,如"pwke"是一個子系列灶体,不是子串。

2. 解題思路

2.1 思路一:滑動窗口

窗口通常是在數(shù)組/字符串中由開始和結(jié)束索引定義的一系列元素的集合掐暮,即[i, j)(左閉蝎抽,右開),而滑動窗口是可以將兩個邊界向某一方向“滑動”的窗口路克。

例如:
我們將[i, j)向右滑動1個元素樟结,則它將變?yōu)?code>[i+1, j+1)(左閉,右開)
HashSet存儲當前處于窗口[i, j)(最初 j = i)中的字符精算,然后我們向右滑動索引j瓢宦,如果它不在HashSet中,我們會繼續(xù)滑動j灰羽,直到s[j]已經(jīng)存在于HashSet中驮履,此時沒有重復(fù)的最長子字符串將會以索引i開頭

HashSet存放當前處于滑動窗口中的字符,初始HashSet為空廉嚼。
窗口的左邊i不動玫镐,右邊j向右滑動,依次遍歷字符:j為指向字符的索引怠噪。
當j = 0時恐似,遍歷字符p,加入HashSet -- [p]舰绘,則目前無重復(fù)字符的最大子串長度maxSubLength = 1
當j = 1時蹂喻,遍歷字符w葱椭,加入HashSet -- [p, w],則maxSubLength = 2

滑動窗口1.png

當j = 2時口四,遍歷字符w孵运,此時HashSet中已包含重復(fù)字符w,
則窗口的左邊i向右滑蔓彩,右邊j不動治笨,HashSet依次刪除索引i所指向的字符,直到HashSet不再包含該字符w赤嚼,
然后重新將遍歷字符w加入HashSet中旷赖,啟動窗口j繼續(xù)右滑。

滑動窗口2.png

窗口的左邊i不動更卒,右邊j繼續(xù)向右滑動等孵,依次遍歷字符:j為指向字符的索引
當j = 3時,遍歷字符k蹂空,加入HashSet -- [w k]
當j = 4時俯萌,遍歷字符e,加入HashSet -- [w k e]上枕,則此時maxSubLength = 3

滑動窗口3.png

當j = 5時咐熙,遍歷字符w,此時HashSet中已包含重復(fù)字符w辨萍,
則窗口的左邊i向右滑棋恼,右邊j不動,HashSet依次刪除索引i所指向的字符锈玉,直到HashSet不再包含該字符w爪飘,
然后重新將遍歷字符w加入HashSet中,啟動窗口j右滑嘲玫。
結(jié)束悦施,則maxSubLength = 3

滑動窗口4.png

代碼實現(xiàn):

public static int lengthOfLongestSubstring(String s) {
    if (s == null || "".equals(s)) {
        return 0;
    }
    int n = s.length();
    Set<Character> set = new HashSet<>();
    int maxLength = 0;
    int i = 0;
    int j = 0;
    while (i < n && j < n) {
        if (!set.contains(s.charAt(j))) {
            set.add(s.charAt(j++));
            maxLength = Math.max(j - i, maxLength);
        } else {
            set.remove(s.charAt(i++));
        }
    }
    return maxLength;
}

2.2 思路二:優(yōu)化的滑動窗口

針對上述的滑動窗口方法,不使用集合來判斷一個字符是否存在去团,而定義字符到索引的映射map抡诞。
當找到重復(fù)字符時,可以立即跳過該窗口土陪。

如果在[i, j)范圍內(nèi)有與s[j]重復(fù)的字符昼汗,索引為j',即[i, ... j', ... j)鬼雀,
我們不需要逐漸增加i顷窒,可以直接跳過[i, j']范圍內(nèi)的所有元素,并將i變?yōu)?code>j' + 1

舉例:
s = "pwwkew",當 j = 2時鞋吉,窗口[0, 2)范圍內(nèi)有與s[2]重復(fù)的字符w鸦做,索引為1(j' = 1),我們可以直接跳過[0, 1]范圍內(nèi)的所有元素谓着,將i變?yōu)?(j' + 1)

代碼實現(xiàn):

public static int lengthOfLongestSubstring(String s) {
  if (s == null || "".equals(s)) {
        return 0;
    }
    int n = s.length();
    Map<Character, Integer> map = new HashMap<>(16);
    int maxLength = 0;
    for (int i = 0,j = 0; j < n; j++) {
        if (map.containsKey(s.charAt(j))) {
            i = Math.max(map.get(s.charAt(j)), i);
        }
        maxLength = Math.max(maxLength, j - i + 1);
        map.put(s.charAt(j), j + 1);
    }
    return maxLength;
}

2.3 思路三:使用列表List存儲不重復(fù)字符

依次遍歷字符串 s 中的每次字符泼诱,若不是重復(fù)字符,加入一個列表list中赊锚,若是重復(fù)字符治筒,則刪除列表list中該重復(fù)字符以及前面的所有字符,然后再將該重復(fù)字符加入此列表list中舷蒲。

在遍歷的過程中統(tǒng)計出列表list中存儲的最大字符個數(shù)耸袜,即為字符串 s 中不含重復(fù)字符的最長子串的長度。

代碼實現(xiàn)

public static int lengthOfLongestSubstring(String s) {
    if (s == null || "".equals(s)) {
        return 0;
    }
    List<Character> list = new ArrayList<>();
    int n = s.length();
    int maxLength = 0;
    for (int i = 0; i < n; i++) {
        int index = list.indexOf(s.charAt(i));
        if (index == -1) {
            list.add(s.charAt(i));
            maxLength = Math.max(list.size(), maxLength);
        } else {
            for (int j = index; j >= 0; j--) {
                list.remove(j);
            }
            list.add(s.charAt(i));
        }
    }
    return maxLength;
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末牲平,一起剝皮案震驚了整個濱河市堤框,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌欠拾,老刑警劉巖胰锌,帶你破解...
    沈念sama閱讀 217,734評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異藐窄,居然都是意外死亡,警方通過查閱死者的電腦和手機酬土,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評論 3 394
  • 文/潘曉璐 我一進店門荆忍,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人撤缴,你說我怎么就攤上這事刹枉。” “怎么了屈呕?”我有些...
    開封第一講書人閱讀 164,133評論 0 354
  • 文/不壞的土叔 我叫張陵微宝,是天一觀的道長。 經(jīng)常有香客問我虎眨,道長蟋软,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,532評論 1 293
  • 正文 為了忘掉前任嗽桩,我火速辦了婚禮岳守,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘碌冶。我一直安慰自己湿痢,他們只是感情好,可當我...
    茶點故事閱讀 67,585評論 6 392
  • 文/花漫 我一把揭開白布扑庞。 她就那樣靜靜地躺著譬重,像睡著了一般拒逮。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上臀规,一...
    開封第一講書人閱讀 51,462評論 1 302
  • 那天滩援,我揣著相機與錄音,去河邊找鬼以现。 笑死狠怨,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的邑遏。 我是一名探鬼主播佣赖,決...
    沈念sama閱讀 40,262評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼记盒!你這毒婦竟也來了憎蛤?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,153評論 0 276
  • 序言:老撾萬榮一對情侶失蹤纪吮,失蹤者是張志新(化名)和其女友劉穎俩檬,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體碾盟,經(jīng)...
    沈念sama閱讀 45,587評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡棚辽,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,792評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了冰肴。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片屈藐。...
    茶點故事閱讀 39,919評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖熙尉,靈堂內(nèi)的尸體忽然破棺而出联逻,到底是詐尸還是另有隱情,我是刑警寧澤检痰,帶...
    沈念sama閱讀 35,635評論 5 345
  • 正文 年R本政府宣布包归,位于F島的核電站,受9級特大地震影響铅歼,放射性物質(zhì)發(fā)生泄漏公壤。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,237評論 3 329
  • 文/蒙蒙 一谭贪、第九天 我趴在偏房一處隱蔽的房頂上張望境钟。 院中可真熱鬧,春花似錦俭识、人聲如沸慨削。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽缚态。三九已至磁椒,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間玫芦,已是汗流浹背浆熔。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留桥帆,地道東北人医增。 一個月前我還...
    沈念sama閱讀 48,048評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像老虫,于是被迫代替她去往敵國和親叶骨。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,864評論 2 354

推薦閱讀更多精彩內(nèi)容