HDU 1266 超级密码 bfs

超级密码

Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)
Total Submission(s): 4293    Accepted Submission(s): 1379


Problem Description
Ignatius花了一个星期的时间终于找到了传说中的宝藏,宝藏被放在一个房间里,房间的门用密码锁起来了,在门旁边的墙上有一些关于密码的提示信息:
密码是一个C进制的数,并且只能由给定的M个数字构成,同时密码是一个给定十进制整数N(0<=N<=5000)的正整数倍(如果存在多个满足条件的数,那么最小的那个就是密码),如果这样的密码存在,那么当你输入它以后门将打开,如果不存在这样的密码......那就把门炸了吧.

注意:由于宝藏的历史久远,当时的系统最多只能保存500位密码.因此如果得到的密码长度大于500也不能用来开启房门,这种情况也被认为密码不存在.
 

Input
输入数据的第一行是一个整数T(1<=T<=300),表示测试数据的数量.每组测试数据的第一行是两个整数N(0<=N<=5000)和C(2<=C<=16),其中N表示的是题目描述中的给定十进制整数,C是密码的进制数.测试数据的第二行是一个整数M(1<=M<=16),它表示构成密码的数字的数量,然后是M个数字用来表示构成密码的数字.两个测试数据之间会有一个空行隔开.

注意:在给出的M个数字中,如果存在超过10的数,我们约定用A来表示10,B来表示11,C来表示12,D来表示13,E来表示14,F来表示15.我保证输入数据都是合法的.
 

Output
对于每组测试数据,如果存在要求的密码,则输出该密码,如果密码不存在,则输出"give me the bomb please".

注意:构成密码的数字不一定全部都要用上;密码有可能非常长,不要试图用一个整型变量来保存密码;我保证密码最高位不为0(除非密码本身就是0).
 

Sample Input
  
  
3 22 10 3 7 0 1 2 10 1 1 25 16 3 A B C
 

Sample Output
  
  
110 give me the bomb please CCB
Hint
Hint
Huge input, scanf is recommended.
 
思路:一道需要细心的题,千万不要想着枚举N整数倍进行匹配,因为密码最多500位,想想2的500次方会是一个多大的数,只能用bfs一个个枚举各种组合的可能,并判断是不是N的整数倍,这里每进行一次求和运算就要对n取模,以防止溢出,并且注意相同余数只能出现一次,因为每个数字都是可以重复出现的,所以把相同余数压入队列只是在做重复计算,所以定义一个标志数组vis[5005]来记录余数出现的情况。。另外还要注意n==0的情况。
代码:
#include<stdio.h>
#include<queue>
#include<algorithm>
#include<string>
#include<string.h>
using namespace std;
int n,c,m;
int num[20];
bool vis[5005];
struct Node
{
	string str;
	int len;
	int sum;
//	Node(){}
//	Node(int *_ans,int _len,int _sum):ans(_ans),len(_len),sum(_sum){}
};
int cmp(int a,int b)
{
	return a<b;
}
char toChar(int a)
{
	if(a<=9)
	  return a+'0';
	return a-10+'A';
}
bool  bfs()
{
	queue<Node> q; 
	Node node;
	for(int i=0;i<m;i++)
	{
		if(num[i]==0) continue;//首位字母不为0 
	    node.str=""; 
	    node.str+=toChar(num[i]);
		node.len=1;
		node.sum=num[i];
		q.push(node); 
	}
//	printf("q.size==%d\n",q.size());
	while(!q.empty())
	{
		node=q.front();
		q.pop();
		int len=node.len;
		int sum=node.sum;
		string str=node.str;
		if(len>500) continue;//长度大于500 ,结束
	//	for(int i=0;i<len;i++) printf("%c",str[i]);
	//	printf("  len==%d  sum==%d\n",len,sum);
		if(sum%n==0) //这里对sum取模,判断num[i]是否满足条件 即num[i]%n==0 
		{
			for(int i=0;i<len;i++)
	 	    {
	 	       printf("%c",str[i]);	
		    }
		   printf("\n");
		   return true;	
		}
//		if(sum>n)//取余防止溢出 
//		{
//			sum=sum%n;
//			if(vis[sum]) continue; //相同余数跳过 
//			vis[sum]=true;//之前没出现过的余数,标记为一 
//		}
		for(int i=0;i<m;i++)
		{
			node.str=str+toChar(num[i]); 
		    node.sum=(sum*c+num[i])%n;//结果模n取余,防止溢出 A=a+b, A%n =>(a%n+b)%n 
		    node.len=len+1; 
		    if(!vis[node.sum])
			{
				vis[node.sum]=true;//之前没出现过的余数,标记为一 
				q.push(node);
			} 
		}
	}
	return false;
	
}
int main()
{
	int T;
	scanf("%d",&T);
	while(T--)
	{
		scanf("%d %d %d",&n,&c,&m);
		getchar(); //接收换行符 
		for(int i=0;i<m;i++)
		{
	       scanf("%c",&num[i]);
		   if(num[i]>='A') num[i]-=55; 
		   else num[i]-='0';
	       getchar();//接收空格 
		}
	 sort(num,num+m,cmp);//优化,从小到大排序 
	 memset(vis,false,sizeof(vis));
	 if(n==0)//特殊情况,n==0 
	 {
	 	if(num[0]==0)
	 	   printf("0\n");
	 	else 
	 	   printf("give me the bomb please\n");
	 }
	 else
	 {
	 	bool flag = bfs();
	 
	    if(!flag)
	    {
	 	printf("give me the bomb please\n");
	    }
	 }
	 
		
//		for(int i=0;i<m;i++)
//		  printf("%d ",num[i]);
	}
}


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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值