HDU -1251 统计难题 【字典树】 题解
目录
- 1.题目
- 2.代码
1.题目
Ignatius最近遇到一个难题,老师交给他很多单词(只有小写字母组成,不会有重复的单词出现),现在老师要他统计出以某个字符串为前缀的单词数量(单词本身也是自己的前缀).
Input
输入数据的第一部分是一张单词表,每行一个单词,单词的长度不超过10,它们代表的是老师交给Ignatius统计的单词,一个空行代表单词表的结束.第二部分是一连串的提问,每行一个提问,每个提问都是一个字符串.
注意:本题只有一组测试数据,处理到文件结束.
Output
对于每个提问,给出以该字符串为前缀的单词的数量.
Sample Input
banana
band
bee
absolute
acm
ba
b
band
abc
Sample Output
2
3
1
0
2.代码
#define _CRT_SECURE_NO_WARNINGS
#include<cstdio>
#include<cstring>
#include<iostream>
using namespace std;
const int maxn = 26;
struct Trie
{
Trie *Next[maxn]; //当前节点可以延伸出的边
int flag;
Trie() //函数,初始化以该信息为前缀的信息个数
{
flag = 1;
memset(Next, NULL, sizeof(Next));
}
}*root; //根节点
void insert(char *str)
{
int len = strlen(str);
Trie *p = root, *q;
for (int i = 0; i < len; i++)
{
int id = str[i] - 'a';
if (p->Next[id] == NULL)如果该节点指向的下一个节点为空(没有一条边延伸出去)
{
q = new Trie(); //新建一个节点
p->Next[id] = q; //存储节点
p = p->Next[id]; ///指向该节点
}
else
{
p = p->Next[id];
++(p->flag); ///累加个数
}
}
}
int search(char *str)
{
int len = strlen(str);
Trie *p = root;
for (int i = 0; i < len; i++)
{
int id = str[i] - 'a';
p = p->Next[id];
if (p == NULL) return 0;
}
return p->flag;
}
///释放
void free(Trie *T)
{
if (T == NULL) return;
for (int i = 0; i < maxn; i++)
{
if (T->Next[i])
{
free(T->Next[i]);
}
}
delete(T);
}
int main()
{
char str[20];
root = new Trie(); //建立根节点
while (gets(str))
{
if (str[0] == '\0') break;
insert(str);
}
while (scanf("%s", str) != EOF)
{
printf("%d\n", search(str));
}
free(root);
return 0;
}
还没有评论,来说两句吧...