统计难题
Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 131070/65535 K (Java/Others)
Total Submission(s): 685 Accepted Submission(s): 295
Problem Description
Ignatius最近遇到一个难题,老师交给他很多单词(只有小写字母组成,不会有重复的单词出现),现在老师要他统计出以某个字符串为前缀的单词数量(单词本身也是自己的前缀).
Input
输入数据的第一部分是一张单词表,每行一个单词,单词的长度不超过10,它们代表的是老师交给Ignatius统计的单词,一个空行代表单词表的结束.第二部分是一连串的提问,每行一个提问,每个提问都是一个字符串.
注意:本题只有一组测试数据,处理到文件结束.
Output
对于每个提问,给出以该字符串为前缀的单词的数量.
Sample Input
banana
band
bee
absolute
acm
ba
b
band
abc
Sample Output
2
3
1
0
Author
Ignatius.L
Recommend
Ignatius.L
/*
这道题编译器需要选用c++才能过,用g++,内存就会超
*/
#include<iostream>
#include<stdio.h>
#include<algorithm>
#include<string>
#include<string.h>
#include<math.h>
#include<map>
#include<set>
#include<string>
#define mod 1000000007
using namespace std;
typedef struct Node
{
bool is;
Node *nxt[27];//26个英文字母
Node()
{
is = false;//is为true,则代表一个单词的结束
memset(nxt,0,sizeof(nxt));
}
}Node;
string word[500005];
int cnt = 0;
void Insert(Node * root,string s)
{
int indx=0;
Node *p = root;
while(indx<s.size())//将单词的每个字母都插入到树中
{
int k = s[indx++] - 'a';
if(p->nxt[k] == 0)
p->nxt[k] = new Node();
p = p->nxt[k];
}
p->is = true;
}
void Cal(Node * p)
{
if(p->is)//如果当前节点是一个单词的结束标记
{
cnt++;
// return;
}
for(int i=0; i<=25; ++i)
{
if(p->nxt[i])//继续往下搜索
Cal(p->nxt[i]);
}
}
void Search(Node *root,string s)
{
Node *p = root;
int indx = 0;
while(indx < s.size())
{
int k = s[indx++] - 'a';
if(p->nxt[k] == 0)
return;
p = p->nxt[k];
}
Cal(p);//从该前缀的最后一个字母出发,查找单词数量
}
int main(void)
{
// freopen("in.txt","r",stdin);
int indx = 0;
Node *root = new Node();
while(1)
{
getline(cin,word[indx]);
if(!word[indx].compare("")) break;//读取到一个空行,就结束读入
Insert(root,word[indx++]);//建树
}
string str;
while(cin >> str)
{
cnt = 0;
Search(root,str);
cout << cnt<<endl;
}
return 0;
}