2797:最短前缀(贪心算法)
- 总时间限制:
- 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
-
1 #include <iostream> 2 #include <string> 3 using namespace std; 4 string s[1005]; 5 int main() 6 { 7 int n=0; 8 while(cin>>s[++n]);//输入完以后 先按回车,再按"CTRL+Z",再按回车 9 for(int i=1; i<=n; ++i)//从第1个字符串数组开始 10 { 11 for(int j=1; j<=s[i].size(); ++j)//长度从1到s[i]的最大长度 12 { 13 //substr功能是复制子字符串,要求从指定位置开始,并具有指定的长度 14 //substr(指定位置,指定的长度) 15 string ss=s[i].substr(0,j); 16 int flag=1; 17 for(int k=1; k<=n; ++k)//从第1个字符串数组开始,一一比较 18 { 19 if(k!=i&&s[k].substr(0,j)==ss)//如果有别的字符串数组的前缀和这个相等,则失败 20 { 21 flag=0; 22 break; 23 } 24 } 25 if(flag||j==s[i].size())//满足条件输出 26 { 27 cout<<s[i]<<" "<<ss<<endl; 28 break; 29 } 30 } 31 } 32 }
越努力越幸运