题目链接:http://noi.openjudge.cn/ch0406/1799/
- 总时间限制: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
- 来源
- 翻译自Rocky Mountain 2004
【思路】
暴力枚举:i枚举字符串,j枚举i串的前缀,k枚举所有字符串判断是否为前缀。时间复杂度O(m*n^2),其中n是字符串个数,m是字符串长度。
假如测试数据真的达到极限:1000*1000*20,这个算法只能得部分测试点的分。
代码如下:
1 #include<stdio.h> 2 #include<string.h> 3 4 char str[1005][23]; 5 6 int main(int argc, char *argv[]) 7 { 8 int n,i,j,k; 9 char temp[30]; 10 int f,len; 11 12 i=0; 13 while(scanf("%s",str[i])!=EOF) i++; 14 15 n=i; 16 for(i=0;i<n;i++) 17 { 18 temp[0]='\0'; 19 len=strlen(str[i]); 20 for(j=1;j<=len;j++) 21 { 22 //strncpy(temp,str[i],j); 23 temp[j-1]=str[i][j-1]; 24 temp[j]='\0'; 25 f=1; 26 for(k=0;k<n;k++) 27 { 28 if(k!=i&&strncmp(temp,str[k],j)==0) { f=0; break; } 29 } 30 if(f==1) { printf("%s %s\n",str[i],temp); break; } 31 } 32 if(j>len) printf("%s %s\n",str[i],str[i]); 33 } 34 return 0; 35 }
网上资料:可以用字典树解决,这个留待以后研究吧。
参考:http://blog.csdn.net/hahalidaxin/article/details/51017502