hdu1195 Open the Lock

    题目大意:有一个紧急开启密码锁的任务。密码由四位数字组成;每个数字从19;每次,可以对每一个数字进行加1或者减1;当从1加到9时,由9再加1会变为1;当从9减到1时,由1再减1会变为9;也可以交换两个相邻的数字,每次操作作为一个step。你的任务就是用最少的步骤解锁!

 

先来分析一下这道题目: (别人分析的,我觉得分析的好,摘抄一下)

  1. 首先考虑1234的下一step有可能是什么样的数字  
  2. 针对每一位数字有四种操作,+1,-1,左交换,右交换(最左边的数字只有右交换,最右边只有左交换)  
  3. 那么下一step的数字  
  4. 第一数字位:2234,9234,2134 
  5. 第二数字位:1334,1134,2134,1324 
  6. 第三数字位:1244,1224,1324,1243 
  7. 第四数字位:1235,1233,1243
    //hdu1195
    #include <stdio.h>
    #include <string.h>
    #include <algorithm>
    #include <queue>
    using namespace std;
    
    typedef struct In  
    {
    	char buf[5];
    	int step;
    }Data;
    
    Data cur, xx;
    queue<Data>q;
    int vist[10][10][10][10];
    char buf1[5], buf2[5];
    
    void bfs()
    {
    	int i;
    	memset(vist, 0, sizeof(vist));
    	while (!q.empty()) q.pop();
    	strcpy(cur.buf, buf1);
    	cur.step = 0;
    	q.push(cur);
    	vist[cur.buf[0]-'0'][cur.buf[1]-'0'][cur.buf[2]-'0'][cur.buf[3]-'0'] = 1;
    	while(!q.empty())
    	{
    		xx = q.front();
    		q.pop();
    		for (i = 0; i <= 3; i++)
    		{
    			strcpy(cur.buf, xx.buf);
    			cur.step = xx.step + 1;
    			if (cur.buf[i] == '9') cur.buf[i] = '1';
    			else cur.buf[i] += 1;
    			if (strcmp(cur.buf, buf2) == 0)
    			{
    				printf("%d\n", cur.step);
    				return;
    			}
    			else if (vist[cur.buf[0]-'0'][cur.buf[1]-'0'][cur.buf[2]-'0'][cur.buf[3]-'0'] != 1)
    			{
    				q.push(cur);
    				vist[cur.buf[0]-'0'][cur.buf[1]-'0'][cur.buf[2]-'0'][cur.buf[3]-'0'] = 1;
    			}
    		}
    		for (i = 0; i <= 3; i++)
    		{
    			strcpy(cur.buf, xx.buf);
    			cur.step = xx.step + 1;
    			if (cur.buf[i] == '1') cur.buf[i] = '9';
    			else cur.buf[i] -= 1;
    			if (strcmp(cur.buf, buf2) == 0)
    			{
    				printf("%d\n", cur.step);
    				return;
    			}
    			else if (vist[cur.buf[0]-'0'][cur.buf[1]-'0'][cur.buf[2]-'0'][cur.buf[3]-'0'] != 1)
    			{
    				q.push(cur);
    				vist[cur.buf[0]-'0'][cur.buf[1]-'0'][cur.buf[2]-'0'][cur.buf[3]-'0'] = 1;
    			}
    		}
    		for (i = 0; i < 3; i++)
    		{
    			strcpy(cur.buf, xx.buf);
    			cur.step = xx.step + 1;
    			char c = cur.buf[i];
    			cur.buf[i] = cur.buf[i+1];
    			cur.buf[i+1] = c;
    			if (vist[cur.buf[0]-'0'][cur.buf[1]-'0'][cur.buf[2]-'0'][cur.buf[3]-'0'] != 1)
    			{
    				if (strcmp(cur.buf, buf2) == 0)
    				{
    					printf("%d\n", cur.step);
    					return;
    				}
    				else
    				{
    					vist[cur.buf[0]-'0'][cur.buf[1]-'0'][cur.buf[2]-'0'][cur.buf[3]-'0'] = 1;
    					q.push(cur);
    				}
    			}
    		}
    	}
    
    }
    
    int main()
    {
    	int cases;
    	scanf("%d", &cases);
    	while (cases--)
    	{
    		scanf("%s%s", buf1, buf2);
    		if (strcmp(buf1, buf2) == 0) printf("0\n");
    		else bfs();
    	}
    	return 0;
    }

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值