アイデア:
(1) 暴力的なトラバーサル
class Solution {
public:
int lengthOfLongestSubstring(string s)
{
string result;
int max=0;
int remax = 0;
bool flag = false;
int len = s.size();
if (s[0] == ' '||s.size()==1)
{
remax = 1;
return remax;
}
else
{
for(int k =0;k<len;k++)
{
flag = false;
result.clear();
if(remax<max)
remax = max;
max = 0;
for (int i = k;i < len;i++)
{
max++;
if (i == k)
{
result.push_back(s[i]);
}
else
{
for (int j = 0;j < result.size();j++)
{
if (s[i] == result[j])
{
flag = true;
continue;
}
}
if (flag == true)
{
max = max - 1;
break;
}
else
result.push_back(s[i]);
}
}
}
}
return remax;
}
};
(2) スライディングウィンドウ
class Solution {
public:
int lengthOfLongestSubstring(string s) {
int res = 0;
int len = s.size();
int freq[256] = {
0 };
int l = 0, r = -1;
while (l < len) {
if (r + 1 < len && freq[s[r + 1]] == 0) {
freq[s[++r]]++;
}
else {
freq[s[l++]]--;
}
res = max(res, r - l + 1);
}
return res;
}
};
スライディング ウィンドウのコードで、2 つの連続する W があるために pwwkew の文字列に遭遇した場合、左ポインターを 2 番目の W に直接移動させたいと考えています。コードによると、最初に左ポインターを移動させることができます。右ポインターを右に直接横切り、右ポインターを上に移動します。
このコードにより、左ポインターが指す W の値が -1 に変更されるため、右ポインターが指すオブジェクトの値は 0 ではなくなり、移動が続行されます。理解するのは難しいです。
(3)
ハッシュテーブルの核となるスライディングウィンドウ+スライディングウィンドウのコードはそのままですが、繰り返し要素の次の要素を左ポインタで指す方法を考える必要があります (これは非常に重要です。多くの問題を引き起こします)。ハッシュ テーブルを使用すると、左側の要素を継続的に削除することで目的を達成できます。
class Solution {
public:
int lengthOfLongestSubstring(string s) {
// 哈希集合,记录每个字符是否出现过
unordered_set<char> occ;
int n = s.size();
// 右指针,初始值为 -1,相当于我们在字符串的左边界的左侧,还没有开始移动
int rk = -1, ans = 0;
// 枚举左指针的位置,初始值隐性地表示为 -1
for (int i = 0; i < n; ++i) {
if (i != 0) {
// 左指针向右移动一格,移除一个字符
occ.erase(s[i - 1]);
}
while (rk + 1 < n && !occ.count(s[rk + 1])) {
// 不断地移动右指针
occ.insert(s[rk + 1]);
++rk;
}
// 第 i 到 rk 个字符是一个极长的无重复字符子串
ans = max(ans, rk - i + 1);
}
return ans;
}
};