poj 2797 最短前缀

2797:最短前缀
Time Limit: 1000ms          

Memory Limit: 
65536kB
Description
一个字符串的前缀是从该字符串的第一个字符起始的一个子串。例如 "carbon"的字串是: "c", "ca", "car", "carb", "carbo", 和 "carbon"。注意到这里我们不认为空串是字串, 但是每个非空串是它自身的字串. 我们现在希望能用前缀来缩略的表示单词。例如, "carbohydrate" 通常用"carb"来缩略表示. 现在给你一组单词, 要求你找到唯一标识每个单词的最短前缀
在下面的例子中,"carbohydrate" 能被缩略成"carboh", 但是不能被缩略成"carbo" (或其余更短的前缀) 因为已经有一个单词用"carbo"开始
一个精确匹配会覆盖一个前缀匹配,例如,前缀"car"精确匹配单词"car". 因此 "car" 是 "car"的缩略语是没有二义性的 , “car”不会被当成"carriage"或者任何在列表中以"car"开始的单词.
Input
输入包括至少2行,至多1000行. 每行包括一个以小写字母组成的单词,单词长度至少是1,至多是20.

Sample Input
carbohydrate
cart
carburetor
caramel
caribou
carbonic
cartilage
carbon
carriage
carton
car
carbonate
Output

输出的行数与输入的行数相同。每行输出由相应行输入的单词开始,后面跟着一个空格接下来是相应单词的没有二义性的最短前缀标识符。
Sample Output

carbohydrate carboh
cart cart
carburetor carbu
caramel cara
caribou cari
carbonic carboni
cartilage carti
carbon carbon
carriage carr
carton carto
car car
carbonate carbona

只需要用一个三次循环来做,大循环为输入的第i个字符串,中层循环为从第一个字符开始长度为1到strlen(i)的子串

小循环为子串与其他的第j个字符串进行比较,

初次做题的时候,采用的strstr(其余字符串,子串)是否存在的作法,提交后为wrong answer,考虑后得知,判断前缀子串是否出现在其他字符串中,需要判断这个子串是否为其他字符串的前缀,而不仅仅判断是否为子串

另外,此种情况判断是否输入结束,可以使用判断

while(scanf("%s",s[count])!=EOF&&source[n][count])或者while(gets(s[count])&&s[count][0])

#include<iostream>
#include<stdio.h>
#include<string.h>
using namespace std;
int main(){
	char s[1002][22];
	char tmp[22];
	int len[1002];
	int i,j,k,count=0;
	while((scanf("%s",s[count]))!=EOF){
		len[count]=strlen(s[count]);
		count++;
	}
	for(i=0;i<count;i++){
		for(j=1;j<=len[i];j++){
			memset(tmp,0,sizeof(tmp));
			for(k=0;k<j;k++)
				tmp[k]=s[i][k];
			tmp[k]='\0';
			//cout<<tmp<<endl;
			for(k=0;k<count;k++){
				if(k!=i){
					if(strstr(s[k],tmp)==s[k])
						break;
				}
			}
			if(k==count){
				cout<<s[i]<<" "<<tmp<<endl;
				break;
			}
		}
		if(j==len[i]+1)
			cout<<s[i]<<" "<<s[i]<<endl;
	}
	return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值