-
总时间限制:
- 1000ms 内存限制:
- 65536kB
-
描述
-
一个字符串的前缀是从该字符串的第一个字符起始的一个子串。例如 "carbon"的字串是: "c", "ca", "car", "carb", "carbo", 和 "carbon"。注意到这里我们不认为空串是字串, 但是每个非空串是它自身的字串. 我们现在希望能用前缀来缩略的表示单词。例如, "carbohydrate" 通常用"carb"来缩略表示. 现在给你一组单词, 要求你找到唯一标识每个单词的最短前缀
在下面的例子中,"carbohydrate" 能被缩略成"carboh", 但是不能被缩略成"carbo" (或其余更短的前缀) 因为已经有一个单词用"carbo"开始
一个精确匹配会覆盖一个前缀匹配,例如,前缀"car"精确匹配单词"car". 因此 "car" 是 "car"的缩略语是没有二义性的 , “car”不会被当成"carriage"或者任何在列表中以"car"开始的单词.
输入
- 输入包括至少2行,至多1000行. 每行包括一个以小写字母组成的单词,单词长度至少是1,至多是20. 输出
- 输出的行数与输入的行数相同。每行输出由相应行输入的单词开始,后面跟着一个空格接下来是相应单词的没有二义性的最短前缀标识符。 样例输入
-
carbohydrate cart carburetor caramel caribou carbonic cartilage carbon carriage carton car carbonate
样例输出
-
carbohydrate carboh cart cart carburetor carbu caramel cara caribou cari carbonic carboni cartilage carti carbon carbon carriage carr carton carto car car carbonate carbona
示例代码(字典树)
-
#include<iostream> #include<string.h> #define SIZE 26 using namespace std; class TrieNode{ public: int num;//有多少单词通过这个节点,即节点字符出现的次数 TrieNode *son[SIZE];//所有的儿子节点的指针 char val;//节点的值 TrieNode():num(1),isEnd(true){ } TrieNode(char v):num(1){ for(int i=0;i<SIZE;++i){ son[i]=NULL; } val=v; } ~TrieNode(){ deleteson(this); } void deleteson(TrieNode* node){ if(node){ for(int i=0;i<SIZE;++i){ if(!node->son[i]){ deleteson(node->son[i]); } delete node->son[i]; } } } }; class Trie{ public: int size; TrieNode root; //建立字典树 void insert(char* str); //查找最短前缀 char* findMinPrefix(char* str); }; void Trie::insert(char* str){ TrieNode *p=&root; for(int i=0;i<strlen(str);++i){ if(!p->son[str[i]-'a']){ p->son[str[i]-'a']=new TrieNode(str[i]); }else{ //经过该节点 (p->son[str[i]-'a'])->num++; } p=p->son[str[i]-'a']; } } char prefix[26]; char* Trie::findMinPrefix(char* str){ TrieNode *p=&root; unsigned int i=0; for(;i<strlen(str);++i){ if(p->son[str[i]-'a']&&(p->son[str[i]-'a'])->num==1){ break; } p=p->son[str[i]-'a']; } strncpy(prefix,str,++i); prefix[i]='\0'; return prefix; } char datas[1000][25]; string s; Trie trie; int main(){ int i=0; while(cin>>datas[i]){ //建立字典树 trie.insert(datas[i]); ++i; } for(int j=0;j<i;++j){ cout<<datas[j]<<" "<<trie.findMinPrefix(datas[j])<<endl; } }