128. 最长连续序列
乍一看感觉很简单,一看要用O(n)???
因为我觉得题目很难而且题目看起来很简单,感觉以后会用到,做个记录
答
:任何一段连续的数都有一个左端点:比如(1,2,3,4)的左端点是1,且找到1之后,发现(1,2,3,4)为最长的连续区间,那么从2,3,4为左端点的区间都不需要继续尝试了,因为他们都比1为左端点的区间短。class Solution
{
public:
int longestConsecutive(vector<int>& nums)
{
int n=nums.size();
unordered_set<int>Hash; //set只有一个参数,然后去重,重复的键值不会被插入
for(int i=0;i<n;i++)
Hash.insert(nums[i]);
int ans=0;
for(int i=0;i<n;i++)
{
if(Hash.find(nums[i]-1)!=Hash.end())//num[i]-1存在,nums[i]不是左端点
continue;
else
{
int x=nums[i]+1; //nums[i]是左端点,从x开始尝试找
int len=1;
while(Hash.find(x)!=Hash.end())
{
x++;
len++;
}
ans=max(ans,len);
}
}
return ans;
}
};
class Solution {
public:
unordered_map<int,int> a;
int find(int x) //找到该元素集合的根节点+路径压缩
{
if(a.count(x))
{
a[x]=find(a[x]);
return a[x];
}
else
return x;
//return a.count(x)?a[x]=find(a[x]):x;
}
int longestConsecutive(vector<int>& nums)
{
for(auto i:nums) //这一步很巧妙,这个错位的赋值使find函数一步到位,只要使连续的,那么就把所有连续的数都放在集合里了。
a[i]=i+1;
int ans=0;
for(auto i:nums)
{
int y=find(i+1);
ans=max(ans,y-i);
}
return ans;
}
};
class Solution
{
public:
int longestConsecutive(vector<int>& nums)
{
unordered_map<int , int>Hash;
int res = 0;
for(int i = 0; i < nums.size(); i++)
{
int now = nums[i];
if(!Hash[now])
{
int left = Hash[now - 1] , right = Hash[now + 1];
int len = left + right + 1;
res = max(res , len);
Hash[now] = len;
Hash[now - left] = len , Hash[now + right] = len;
}
}
return res;
}
};