洛谷 P3879 [TJOI2010]【阅读理解】

题目描述

英语老师留了N篇阅读理解作业,但是每篇英文短文都有很多生词需要查字典,为了节约时间,现在要做个统计,算一算某些生词都在哪几篇短文中出现过。

输入输出格式

输入格式:

第一行为整数N,表示短文篇数,其中每篇短文只含空格和小写字母。

按下来的N行,每行描述一篇短文。每行的开头是一个整数L,表示这篇短文由L个单词组成。接下来是L个单词,单词之间用一个空格分隔。

然后为一个整数M,表示要做几次询问。后面有M行,每行表示一个要统计的生词。

输出格式:

对于每个生词输出一行,统计其在哪几篇短文中出现过,并按从小到大输出短文的序号,序号不应有重复,序号之间用一个空格隔开(注意第一个序号的前面和最后一个序号的后面不应有空格)。如果该单词一直没出现过,则输出一个空行。

输入输出样例

输入样例#1:

3
9 you are a good boy ha ha o yeah
13 o my god you like bleach naruto one piece and so do i
11 but i do not think you will get all the points
5
you
i
o
all
naruto

输出样例#1:

1 2 3
2 3
1 2
3
2

解题思路

  首先就是字典树了,只是我们定一个bool数组判断某个单词是否在某篇文章出现过,然后.........最后一个点就WA了(当时我也蒙了)原来,我们定义bool数组可以换成bitset,这样时间复杂度可以/32,非常实用,强烈安利。

题解

 1 #include<bits/stdc++.h>
 2 #define N 300001
 3 using namespace std;
 4 int n,m,l; 
 5 char x[10001];
 6 struct Trie{
 7     int sz;
 8     int ch[N][26];
 9     bitset<1001>val[N];//bitset 
10     Trie(){
11         sz=1;
12         memset(ch[0],0,sizeof(ch[0]));
13         memset(val,0,sizeof(val));
14     }//初始化 
15     int idx(char c)//字符转数字 
16     {
17         return c-'a';
18     }
19     void insert(string s,int num)//插入 
20     {
21         int u=0;
22         for(int i=0;i<s.size();i++)
23         {
24             int c=idx(s[i]);
25             if(!ch[u][c])
26             {
27                 memset(ch[sz],0,sizeof(ch[sz]));
28                 ch[u][c]=sz;
29                 sz++;
30             }
31             u=ch[u][c];//向下便利 
32         }
33         val[u][num]=1;//在第num篇文章出现过 
34     }
35     void find(string s)//查找 
36     {
37         int u=0;
38         for(int i=0;i<s.size();i++)
39         {
40             int c=idx(s[i]);
41             if(!ch[u][c])//没找到了 
42             {
43                 cout<<endl;
44                 return;
45             }
46             u=ch[u][c];
47         }
48         for(int i=1;i<=n;i++)//看在哪几篇文章出现过 
49         {
50             if(val[u][i]==1)cout<<i<<" ";
51         }
52         cout<<endl;
53     }
54 }tree;
55 int main()
56 {
57     scanf("%d",&n);
58     for(int i=1;i<=n;i++)
59     {
60         cin>>l;
61         for(int j=1;j<=l;j++)
62         {
63             scanf("%s",x);
64             tree.insert(x,i);
65         }
66     }
67     scanf("%d",&m);
68     for(int i=1;i<=m;i++)
69     {
70         scanf("%s",x);
71         tree.find(x);
72     }
73     return 0;
74 }
原文地址:https://www.cnblogs.com/hualian/p/11206745.html