灵梦有n个单词想要背,但她想通过一篇文章中的一段来记住这些单词。
文章由m个单词构成,她想在文章中找出连续的一段,其中包含最多的她想要背的单词(重复的只算一个)。并且在背诵的单词量尽量多的情况下,还要使选出的文章段落尽量短,这样她就可以用尽量短的时间学习尽可能多的单词了。
第1行一个数n,
接下来n行每行是一个长度不超过10的字符串,表示一个要背的单词。
接着是一个数m,
然后是m行长度不超过10的字符串,每个表示文章中的一个单词。
输出文件共2行。第1行为文章中最多包含的要背的单词数,第2行表示在文章中包含最多要背单词的最短的连续段的长度。
3
hot
dog
milk
5
hot
dog
dog
milk
hot
3
3
对于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; }
wikioi 3031 字符串哗然并匹配查找,布布扣,bubuko.com
原文地址:http://blog.csdn.net/u011466175/article/details/38018983