hdu 3937

 http://acm.hdu.edu.cn/showproblem.php?pid=3937

中文描述,题意就没问题了。也没什么好说的,字典树,判断条形码所表示的字符得思考下

直接贴代码

#include<stdio.h>
#include<string.h>
#include<math.h>
#include<algorithm>
using namespace std;
#define ep 1e-8
const int kind=64;
#define maxn 3000550
int tree[maxn][kind],top;
void init()
{
	top=1;
	memset(tree[0],0,sizeof(tree[0]));
}
void insert(char *s)
{
	int rt,nxt,i=0,k;
	for(rt=0;s[i];rt=nxt,i++)
	{
		if(s[i]>='A'&&s[i]<='Z')
			k=s[i]-'A';
		else if(s[i]>='a'&&s[i]<='z')
			k=s[i]-'a'+26;
		else
			k=s[i]-'0'+52;
		nxt=tree[rt][k];
		if(nxt==0)
		{
			tree[rt][k]=nxt=top;
			memset(tree[top],0,sizeof(tree[top]));
			tree[nxt][kind-1]=1;
			top++;
		}
		else
			tree[nxt][kind-1]++;
	}
}
int search(char *s,int ss,int tt)
{
	int rt=0,i,k;
	for(i=ss;i<=tt;i++)
	{
		if(s[i]>='A'&&s[i]<='Z')
			k=s[i]-'A';
		else if(s[i]>='a'&&s[i]<='z')
			k=s[i]-'a'+26;
		else
			k=s[i]-'0'+52;
		if(tree[rt][k]==0) return 0;
		rt=tree[rt][k];
	}
	return tree[rt][kind-1];
}
//判断条形码所表示的字符
struct node
{
	int bit;
	double d;
}a[8];
bool cmp(node a,node b)
{
	return a.d<b.d;
}
int check(node a[])
{
	sort(a,a+8,cmp);
	double Max=a[0].d/0.95; //白色条形码的最小宽度
	double Min=a[7].d/2.10; //白色条形码的最大宽度
	if(Min>Max) return -1;
	double l=Min*0.85;      //白色条形码与黑色条形码的最小宽度差
	for(int i=1;i<8;i++)
	{
		double dis=a[i].d-a[i-1].d;
		if((dis>l||fabs(dis-l)<ep))
			return i;
	}
	return 0;
}
int main()
{
	int n,m,N,i,j,k,flag,p,r;
	char str[32],s[40];
	char begin[]={'b','e','g','i','n'};
	char end[]={'e','n','d'};
	while(scanf("%d%d",&n,&m)!=EOF)
	{
		init();
		for(i=0;i<n;i++)
		{
			scanf("%s",str);
			insert(str);
		}
		for(k=0;k<m;k++)
		{
			scanf("%d",&N);
			memset(s,0,sizeof(s));
			flag=0;
			for(i=0;i<N;i++)
			{
				for(j=0;j<8;j++)
				{
					scanf("%lf",&a[j].d);
					a[j].bit=7-j;
				}
				if(flag||N<=8) continue;
				if((r=check(a))==-1)
					flag=1;
				else
				{
					for(p=r;p<8;p++)
						s[i]+=(1<<a[p].bit);
					if(i<5&&s[i]!=begin[i])
						flag=1;
					if(i>=N-3&&s[i]!=end[i-N+3])
						flag=1;
					if(!((s[i]>='0'&&s[i]<='9')||(s[i]>='a'&&s[i]<='z')||(s[i]>='A'&&s[i]<='Z')))
						flag=1;
				}
			}
			if(flag||N<=8)
				printf("wrong barcode!\n");
			else
			{
				int ans=search(s,5,N-4);
				printf("%d\n",ans);
			}
		}
	}
	return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值