[题解]剑指 Offer 48. 最长不含重复字符的子字符串 (C++)

题目

请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子字符串的长度。

示例 1:

输入: "abcabcbb"
输出: 3 
解释: 因为无重复字符的最长子串是 "abc",所以其长度为 3。

示例 2:

输入: "bbbbb"
输出: 1
解释: 因为无重复字符的最长子串是 "b",所以其长度为 1。

示例 3:

输入: "pwwkew"
输出: 3
解释: 因为无重复字符的最长子串是 "wke",所以其长度为 3。
     请注意,你的答案必须是 子串 的长度,"pwke" 是一个子序列,不是子串。

提示:

s.length <= 40000

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/zui-chang-bu-han-zhong-fu-zi-fu-de-zi-zi-fu-chuan-lcof
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

思路

直接暴力的时间复杂度肯定超了,显然要用动态规划。设dp[i]是以第i个字符为结尾的最长子串的长度,从dp[i-1]到dp[i]的转移公式要怎么写?我们已经知道了dp[i-1],那么只需要确认s[i]上一次出现的位置j,如果i - j > dp[i - 1],也就是说以第i-1个字符为结尾的最长的子串中不包含s[j](即s[i])这个字符,dp[i] = dp[i-1] + 1就可以成立;如果i - j <= dp[i - 1],那就是说前面的子串包括s[j]在内,则dp[i] = i - j;转移方程是:

dp[i]={dp[i1]+1,ij>dp[i1]ij,ij<=dp[i1]

那么,要怎么确定j呢?可以用一个哈希表来存储每个字符最后一次出现的位置,每次遍历字符的时候更新;这里有一个trick,对于哈希表里没有存过的字符,可以令j为-1,这样就能保证i - j一定大于dp[i - 1]。
时间复杂度和空间复杂度都是O(n)。

代码

class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        int n = s.size();
        if(n < 2) return n;
        vector<int> dp(n, 0);
        unordered_map<char, int> mp;
        dp[0] = 1;
        mp[s[0]] = 0;
        for(int i = 1; i < n; ++i)
        {
            int j = mp.count(s[i]) ? mp[s[i]] : -1;
            mp[s[i]] = i;
            if(dp[i - 1] < i - j)
            {
                dp[i] = dp[i - 1] + 1;
            }
            else
            {
                dp[i] = i - j;
            }
        }
        return *max_element(dp.begin(),  dp.end());
    }
};

改进

仔细考虑更新dp数组的过程,会发现在更新dp数组时,只需要dp[i-1],那么可以只用一个数tmp来存dp[i]的信息,同时每次更新tmp后都更新一下最长子串长度ans,空间复杂度降低到O(1),因为哈希表要额外占用的空间也是有限的(最大就是所有字符,O(128)=O(1))。

代码

class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        unordered_map<char, int> mp;
        int tmp = 0, ans = 0;
        for(int i = 0; i < s.size(); ++i)
        {
            int j = mp.count(s[i]) ? mp[s[i]] : -1;
            mp[s[i]] = i;
            if(tmp < i - j)
            {
                ++tmp;
            }
            else
            {
                tmp = i - j;
            }
            ans = max(ans, tmp);
        }
        return ans;
    }
};
posted @   浮生的刹那  阅读(80)  评论(0编辑  收藏  举报
编辑推荐:
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
阅读排行:
· 全程不用写代码,我用AI程序员写了一个飞机大战
· DeepSeek 开源周回顾「GitHub 热点速览」
· 记一次.NET内存居高不下排查解决与启示
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
  1. 1 Night City r e l,Artemis Delta
  2. 2 Gold Steps(人生何处不青山) Neck Deep
  3. 3 Devil Trigger Ali Edwards
  4. 4 Hopeless Case Roam
  5. 5 On My Own Blitz Kids
  6. 6 I Really Want to Stay At Your House Rosa Walton & Hallie Coggins
  7. 7 Major Crimes Health & Window Weather
Gold Steps(人生何处不青山) - Neck Deep
00:00 / 00:00
An audio error has occurred, player will skip forward in 2 seconds.
点击右上角即可分享
微信分享提示