题目:
找球号(二)
时间限制: 1000 ms | 内存限制: 65535 KB
难度: 5
-
-
描述
-
在某一国度里流行着一种游戏。游戏规则为:现有一堆球中,每个球上都有一个整数编号i(0<=i<=100000000),编号可重复,还有一个空箱子,现在有两种动作:一种是"ADD",表示向空箱子里放m(0<m<=100)个球,另一种是"QUERY”,表示说出M(0<M<=100)个随机整数ki(0<=ki<=100000100),分别判断编号为ki 的球是否在这个空箱子中(存在为"YES",否则为"NO"),先答出者为胜。现在有一个人想玩玩这个游戏,但他又很懒。他希望你能帮助他取得胜利。
-
-
输入
-
第一行有一个整数n(0<n<=10000);
随后有n行;
每行可能出现如下的任意一种形式:
第一种:
一个字符串"ADD",接着是一个整数m,随后有m个i;
第二种:
一个字符串"QUERY”,接着是一个整数M,随后有M个ki;
\ -
输出
-
输出每次询问的结果"YES"或"NO".
-
样例输入
-
2 ADD 5 34 343 54 6 2 QUERY 4 34 54 33 66 -
样例输出
-
YES YES NO NO
-
-
-
代码1( 未AC,但是代码正确,二分法):
#include <stdio.h>
#include <iostream>
#include <algorithm>
#include <string.h>
using namespace std;
int a[10000001];
int erfen(int index,int m)
{
int mid;
int start=0,end=m-1;
while(start<=end)
{
mid=(start+end)/2;
if(a[mid]>index)
{
end=mid-1;
}
else if(a[mid]<index)
{
start=mid+1;
}
else
return 1;
}
return 0;
}
int main()
{
int n;
int x=0;
scanf("%d",&n);
char b[15];
while(n--)
{
memset(b,'\0',sizeof(b));
scanf("%s",b);
if(strcmp(b,"ADD")==0)
{
int m;
scanf("%d",&m);
for(int i=0; i<m; i++)
scanf("%d",&a[x++]);
}
else if(strcmp(b,"QUERY")==0)
{
int k;
scanf("%d",&k);
sort(a,a+x);
while(k--)
{
int m;
scanf("%d",&m);
printf("%s\n",erfen(m,x)?"YES":"NO");
}
}
}
return 0;
}
代码2( vector容器/ac):
#include <stdio.h>
#include <iostream>
#include <algorithm>
#include <string.h>
#include <vector>
using namespace std;
vector<bool>a(100000001);
int main()
{
int n;
int x=0;
scanf("%d",&n);
char b[15];
while(n--)
{
memset(b,'\0',sizeof(b));
scanf("%s",b);
if(strcmp(b,"ADD")==0)
{
int m,n;
scanf("%d",&m);
while(m--)
{
scanf("%d",&n);
a[n]=true;
}
}
else if(strcmp(b,"QUERY")==0)
{
int k;
scanf("%d",&k);
while(k--)
{
int m;
scanf("%d",&m);
printf("%s\n",a[m]==true?"YES":"NO");
}
}
}
return 0;
}
代码3( 哈希)://代码来自于别人
#include<iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<time.h>
#include<algorithm>
using namespace std;
#define N 1000010
#define CLR(arr, what) memset(arr, what, sizeof(arr))
const int fib = 111123;
int Key[N], Head[N], Next[N];
int top;
void add(int n)
{
int temp;
temp = n % fib;
Key[top] = n;
Next[top] = Head[temp];
Head[temp] = top;
top++;
}
int main()
{
int ncase;
char str[8];
int num, number;
bool flag;
CLR(Key, 0);
CLR(Head, -1);
top = 0;
flag = false;
scanf("%d", &ncase);
while(ncase--)
{
scanf("%s", str);
if(str[0] == 'A')
{
scanf("%d", &num);
for(int i = 0; i < num; ++i)
{
scanf("%d", &number);
add(number);
}
}
else
{
scanf("%d", &num);
for(int i = 0; i < num; ++i)
{
scanf("%d", &number);
int temp = number % fib;
for(int j = Head[temp]; j != -1; j = Next[j])
if(Key[j] == number)
{
flag = true;
break;
}
printf(flag == true ? "YES\n" : "NO\n");
flag = false;
}
}
}
return 0;
}
ps: 虽然名字很高大上,但是这种方法我并不会。。。以后会了再补
代码4( 位运算版)://代码来自于别人
#include<stdio.h>
unsigned int hash[3125005] = {0};
int main()
{
int i,a,t,n;
char str[10];
scanf("%d", &t);
while(t--)
{
scanf("%s %d", str, &n);
if(str[0] == 'A')
{
for(i = 0; i < n; ++i)
{
scanf("%d", &a);
hash[a / 32] |= 1 << (a % 32);
}
}
else if(str[0] == 'Q')
{
for(i = 0; i < n; ++i)
{
scanf("%d", &a);
if(hash[a / 32] & (1 << (a % 32)))
printf("YES\n");
else
printf("NO\n");
}
}
}
return 0;
}
ps: 3125005怎么来的?因为最大值是10^7+10。用32来除法散列,(10^7+10)/32 ~~3125000.3125,所以取3125005。为什么是用32来散列呢?数据说了数值不会超过32位。虽然是这样,但是这种方法还是很难理解
\