首页 > 代码库 > wikioi 3031 字符串哗然并匹配查找

wikioi 3031 字符串哗然并匹配查找

题目描述 Description

灵梦有n个单词想要背,但她想通过一篇文章中的一段来记住这些单词。

    文章由m个单词构成,她想在文章中找出连续的一段,其中包含最多的她想要背的单词(重复的只算一个)。并且在背诵的单词量尽量多的情况下,还要使选出的文章段落尽量短,这样她就可以用尽量短的时间学习尽可能多的单词了。

输入描述 Input Description

第1行一个数n,

接下来n行每行是一个长度不超过10的字符串,表示一个要背的单词。

接着是一个数m,

然后是m行长度不超过10的字符串,每个表示文章中的一个单词。

输出描述 Output Description

输出文件共2行。第1行为文章中最多包含的要背的单词数,第2行表示在文章中包含最多要背单词的最短的连续段的长度。

 

样例输入 Sample Input

3

hot

dog

milk

5

hot

dog

dog

milk

hot

样例输出 Sample Output

3

3

数据范围及提示 Data Size & Hint

对于30%的数据 n<=50,m<=500;

对于60%的数据 n<=300,m<=5000;

对于100%的数据 n<=1000,m<=100000;


这题WA了三十多发,实在是无语死了。

刚开始写的l<m,然后没发现,几乎把所有的代码都动了一遍就是没改这里,然后都是只得90分,第六组数据没过。

原本以为是哈希值有冲突了,然后把模改成其他的素数了还是不行,然后把哈希方法改成其他的了还是不行,后面只能投降了。

实在不行了,然后把数据下载下来,运行才发现输出负数了。

唉……然后加了句if判断输出负数的就输出0.

后面过了才知道是这里错了。

这题搞了二天了都。对字符串哗然印象深刻了这次。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<map>
#include<bitset>
#define INF 100007
using namespace std;
typedef unsigned long long ull;
map<ull,int>vis,Map;
map<int,ull>b;
ull hash(char *s)
{
    int i,len=strlen(s);
    ull sum=0;
    for(i=0;i<len;i++)
        sum=sum*131+s[i]-'a';
    return sum;
}
int main()
{
    int n,m,i,sum=0;
    char a[12];
    cin>>n;
    for(i=0;i<n;i++)
    {
        scanf("%s",a);
        Map[hash(a)]=1;
    }
    cin>>m;
    for(i=0;i<m;i++)
    {
        scanf("%s",a);
        b[i]=hash(a);
        if(Map[b[i]]&&!vis[b[i]]) sum++,vis[b[i]]=1;
    }
    vis.clear();
    printf("%d\n",sum);
    int l=0,r=0,sum1=0,Min=0x7fffffff;
    while(r<m)
    {
        while(r<m)
        {
            if(Map[b[r]]&&!vis[b[r]]) sum1++;
            vis[b[r]]++;r++;
            if(sum1==sum) break;
        }
        while(l<r&&(!Map[b[l]]||vis[b[l]]>1))
        {
            vis[b[l]]--;
            l++;
        }
        Min=min(Min,r-l);
    }
    //if(Min<0||Min==0x7fffffff) Min=0;
    printf("%d\n",Min);
    return 0;
}