题目描述 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
第一问很好求, 用map统计就可以了, 第二问用双指针的思想。 r一直向右移动, 当map里的个数和第一问求出来的cnt相同的时候, 移动左指针直到刚刚好满足要求。ans= min(ans, r-l), 然后一直这样找下去直到r==m。
#include <iostream> #include <vector> #include <cstdio> #include <cstring> #include <algorithm> #include <cmath> #include <map> #include <set> #include <string> #include <queue> #include <stack> #include <bitset> using namespace std; #define pb(x) push_back(x) #define ll long long #define mk(x, y) make_pair(x, y) #define lson l, m, rt<<1 #define mem(a) memset(a, 0, sizeof(a)) #define rson m+1, r, rt<<1|1 #define mem1(a) memset(a, -1, sizeof(a)) #define mem2(a) memset(a, 0x3f, sizeof(a)) #define rep(i, n, a) for(int i = a; i<n; i++) #define fi first #define se second typedef pair<int, int> pll; const double PI = acos(-1.0); const double eps = 1e-8; const int mod = 1e9+7; const int inf = 1061109567; const int dir[][2] = { {-1, 0}, {1, 0}, {0, -1}, {0, 1} }; map <string, int> mp1, mp2; string a[100005]; int main() { int n, m, cnt = 0, tot = 0; string s; cin>>n; for(int i = 0; i<n; i++) { cin>>s; mp1[s] = 1; } cin>>m; for(int i = 0; i<m; i++) { cin>>a[i]; if(mp1[a[i]]&&!mp2[a[i]]) { cnt++; mp2[a[i]] = 1; } } mp2.clear(); cout<<cnt<<endl; int r = 0, l = 0, ans = inf; while(r<m) { while(r<m) { if(!mp1[a[r]]) { r++; continue; } if(!mp2[a[r]]) { tot++; } mp2[a[r]]++; r++; if(tot == cnt) break; } while(l<r && (!mp2[a[l]] || mp2[a[l]]>1)) { if(mp2[a[l]]) mp2[a[l]]--; l++; } ans = min(ans, r-l); } cout<<ans<<endl; return 0; }
时间: 2024-10-27 02:11:28