http://acm.dlut.edu.cn/problem.php?id=1247
就是一道看起来很简单的题!
每次操作可以插入一些数,查询一个数是否在集合中,清空集合。但是数据量超级大,要插入查找清空都是o(1)的复杂度才可以,不可以用stl的set,hashtable也不可以用vector...所以用链式向前星的办法。
不过链式向前星插入和查找都是o(1)的算法,但是用memset清空却是o(n)的算法,所以我们可以再开一个数组优化。设置一个版本标记,初始为零。每次清空把版本标记自加。查找和插入的时候都要判断当前节点的版本是否符合当前版本。不符合就把当前节点清零,再把当前节点版本更新到最新,再进行别的操作。
#include<iostream>
#include<cstdio>
using namespace std;
const int mod = 2001000;
struct xx{
int next,n;
}side[mod*10];
int head[mod],v[mod]={0};
inline int abs(int x){
return x&0x7fffffff;
}
int step=0,top=0;
struct hashtable{
inline int hash(int x){
return abs(x)%mod;
}
inline void insert(int x){
int value=hash(x);
if(v[value]!=step){
head[value]=-1;
v[value]=step;
}
side[top]=(xx){head[value],x};
head[value]=top++;
}
inline bool find(int x){
int value=hash(x);
if(v[value]!=step)
head[value]=-1;
for(int i=head[value];i!=-1;i=side[i].next){
if(side[i].n==x) return 1;
}
return 0;
}
inline void clear(){
top=0;
step++;
}
}table;
int main()
{
int T;
scanf("%d",&T);
while(T--){
table.clear();
int N;
scanf("%d",&N);
while(N--){
int a;
scanf("%d",&a);
if(a==1){
int x,k;
scanf("%d%d",&x,&k);
for(k--;k>=0;k--){
table.insert(x+k*2);
}
}
else if(a==2){
int x;
scanf("%d",&x);
if(table.find(x)) printf("Yes\n");
else printf("No\n");
}
else if(a==3){
table.clear();
}
}
}
return 0;
}
#include<cstdio>
using namespace std;
const int mod = 2001000;
struct xx{
int next,n;
}side[mod*10];
int head[mod],v[mod]={0};
inline int abs(int x){
return x&0x7fffffff;
}
int step=0,top=0;
struct hashtable{
inline int hash(int x){
return abs(x)%mod;
}
inline void insert(int x){
int value=hash(x);
if(v[value]!=step){
head[value]=-1;
v[value]=step;
}
side[top]=(xx){head[value],x};
head[value]=top++;
}
inline bool find(int x){
int value=hash(x);
if(v[value]!=step)
head[value]=-1;
for(int i=head[value];i!=-1;i=side[i].next){
if(side[i].n==x) return 1;
}
return 0;
}
inline void clear(){
top=0;
step++;
}
}table;
int main()
{
int T;
scanf("%d",&T);
while(T--){
table.clear();
int N;
scanf("%d",&N);
while(N--){
int a;
scanf("%d",&a);
if(a==1){
int x,k;
scanf("%d%d",&x,&k);
for(k--;k>=0;k--){
table.insert(x+k*2);
}
}
else if(a==2){
int x;
scanf("%d",&x);
if(table.find(x)) printf("Yes\n");
else printf("No\n");
}
else if(a==3){
table.clear();
}
}
}
return 0;
}