hdu1430 魔板(康拓展开 bfs预处理)
魔板
Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)
Total Submission(s): 4050    Accepted Submission(s): 951
1 2 3 4
8 7 6 5
对于魔板,可施加三种不同的操作,具体操作方法如下:
A: 上下两行互换,如上图可变换为状态87654321
B: 每行同时循环右移一格,如上图可变换为41236785
C: 中间4个方块顺时针旋转一格,如上图可变换为17245368
给你魔板的初始状态与目标状态,请给出由初态到目态变换数最少的变换步骤,若有多种变换方案则取字典序最小的那种。
17245368
12345678
82754631
AC
思路:是先了解了康托展开再做这道题,想当于是用康拓展开给每一种状态弄一个独立的标号,可以快速找到某一种状态,所以就是先用12345678为初始状态,然后记录每一种状态的contor值,这里我用了把函数写在结构体里的方法,方便简洁(学习了网上一位大佬),然后比较难想就是,每一次输入的初态和末态并不都是12345678,如何将输入和bfs中的状态对应起来呢?
我们就需要置换一下。
置换: 原始态为“12345678” 例如 初态是 “45781236” 目态是 ”78451326“
目态的第一位‘7’ 在初态中是第三位, 而原始态的第三位是‘3’,故目态的第一位应该转换为‘3’,(就是将输入的值和12345678一一对应)就这样一次转换 最后目态转换为 34125768, 最后找出34125768对应的康拓值,输出就好了。
而字典序的话,每一次bfs都是按照ABC这样的顺序的,所以每种状态的步骤必然是最小字典序。
具体看代码。
#include<iostream>
#include<algorithm>
#include<cstdlib>
#include<sstream>
#include<cstring>
#include<bitset>
#include<cstdio>
#include<string>
#include<deque>
#include<stack>
#include<cmath>
#include<queue>
#include<set>
#include<map>
#define INF 0x3f3f3f3f
#define CLR(x,y) memset(x,y,sizeof(x))
#define LC(x) (x<<1)
#define RC(x) ((x<<1)+1)
#define MID(x,y) ((x+y)>>1)
using namespace std;
typedef long long ll;
const int N=40320+10;
int fact[9]= {1,1,2,6,24,120,720,5040,40320};
struct info{
	int arr[8];
	int val;//计算出来的康托展开的序号
	string step;
	void contor(){//把康托展开写在结构体里面
		val=0;
		for(int i=0;i<8;i++){
			int t=0;
			for(int j=i+1;j<8;j++){
				if(arr[j]<arr[i])t++;
			}
			val+=t*fact[8-i-1];
		}
		val++;
	}
	void change_a(){
		reverse(arr,arr+8);
		step+='A';
	}
	void change_b(){
        int temp=arr[3];
        for (int i=3; i>0; --i)
            arr[i]=arr[i-1];
        arr[0]=temp;
        temp=arr[4];
        for (int i=4; i<7; ++i)
            arr[i]=arr[i+1];
        arr[7]=temp;
        step+="B";
    }
	void change_c(){
		int temp=arr[1];
		arr[1]=arr[6];
		arr[6]=arr[5];
		arr[5]=arr[2];
		arr[2]=temp;
		step+='C';
	}
};
int vis[N];
string ans[N];//记录每一种状态的操作
int re[8];
info s,t;
void bfs(){
	queue<info>q;
	vis[s.val]=1;
	q.push(s);
	info now,v;
	while(!q.empty()){
		now=q.front();
		q.pop();
		v=now;
		int len=v.step.length();
		if(len<1||v.step[len-1]!='A'){//去掉AA这样的情况
			v.change_a();
			v.contor();
			if(!vis[v.val]){
				vis[v.val]=1;
				ans[v.val]=v.step;
				q.push(v);
			}
		}
		v=now;
		if(len<3||v.step[len-1]!='B'||v.step[len-2]!='B'||v.step[len-3]!='B'){//去掉BBBB这样的情况
			v.change_b();
			v.contor();
			if(!vis[v.val]){
				vis[v.val]=1;
				ans[v.val]=v.step;
				q.push(v);
			}
		}
		v=now;
		if(len<3||v.step[len-1]!='C'||v.step[len-2]!='C'||v.step[len-3]!='C'){//去掉CCCC这样的情况
			v.change_c();
			v.contor();
			if(!vis[v.val]){
				vis[v.val]=1;
				ans[v.val]=v.step;
				q.push(v);
			}
		}
	}
}
int main(){
	for(int i=0;i<8;i++){
		s.arr[i]=i+1;
	}//初始化  12345678
	s.contor();
	bfs();
	char ss[10];
	while(scanf("%s",ss)!=EOF){
		for(int i=0;i<8;i++){
		re[ss[i]-'0']=i+1;
		}//将输入的初始状态的字符 和12345678(bfs中的初始状态)一一对应
		scanf("%s",ss);
		for(int i=0;i<8;i++){
			t.arr[i]=re[ss[i]-'0'];//找到当前的目标状态在bfs中对应的值
		}
		t.contor();
		printf("%s\n",ans[t.val].c_str());
	}
}
 												
											hdu1430 魔板(康拓展开 bfs预处理)的更多相关文章
- hdu-1430 魔板 康拓展开+映射优化
		
给定三种操作,将排列A转化为排列B,求最少步骤. 这种题目可以只跑一次bfs,比如只跑"12345678",那么如果遇到"23456781"->某个字符串 ...
 - HDU 1430 魔板(康托展开+BFS+预处理)
		
魔板 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submis ...
 - hdu1430魔板(BFS+康托展开)
		
做这题先看:http://blog.csdn.net/u010372095/article/details/9904497 Problem Description 在魔方风靡全球之后不久,Rubik先 ...
 - hdu1430魔板
		
Problem Description 在魔方风靡全球之后不久,Rubik先生发明了它的简化版——魔板.魔板由8个同样大小的方块组成,每个方块颜色均不相同,可用数字1-8分别表示.任一时刻魔板的状态可 ...
 - 【HDOJ1043】【康拓展开+BFS】
		
http://acm.hdu.edu.cn/showproblem.php?pid=1043 Eight Time Limit: 10000/5000 MS (Java/Others) Memo ...
 - HDU1430;魔板(BFS+康托展开)
		
传送门 题意 给出初始序列与终止序列,给出三种操作,问最少经过几次操作能使初始->终止,输出操作(字典序最小) 分析 字符串只有8个字符,使用康托展开. 1.BFS将所有序列从"123 ...
 - HDU 1043 & POJ 1077 Eight(康托展开+BFS+预处理)
		
Eight Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 30176 Accepted: 13119 Special ...
 - 洛谷 - P2730 - 魔板 Magic Squares - bfs
		
写状态转移弄了很久,老了,不记得自己的数组是怎么标号的了. #include <bits/stdc++.h> using namespace std; #define ll long lo ...
 - hdu 1430 魔板  康托展开 + 很好的映射
		
http://acm.hdu.edu.cn/showproblem.php?pid=1430 如果从start ---> end,每一次都bfs进行,那么就肯定会超时. 考虑到先把start映射 ...
 
随机推荐
- 关于taskaffinity属性的作用
			
意味着这activity更喜欢哪个TESK,具体见下方说明 当一个包含FLAG_ACTIVITY_NEW_TASK标志的intent启动一个activity时. 一个新的activity,默认地启动到 ...
 - php中用大括号把?>和<?php框起来的作用
			
<?php function my_function() { ?> My function was called <!--就是这里,为什么前面要用?>和< ?php 把M ...
 - CoreData的增删改查
			
首先使用CoreData创建Demo,勾上CoreData选项 然后创建Entity对象,点击Add Entity(+)按钮 生成Entity对象 重命名双击Entity选项,然后输入Person 设 ...
 - 【摘自张宴的"实战:Nginx"】使用nginx的proxy_cache模块替代squid,缓存静态文件
			
#user nobody;worker_processes 1; error_log logs/static_source.error.log;#error_log logs/error.log no ...
 - hibernate 对象OID
			
它是hibernate用于区分两个对象是否是同一个对象的标识. 我们都知道,虚拟机内存区分两个对象看的是内存的地址是否一致.数据库区分两个对象,靠的是表的主键.hibernate负责把内存中的对象持久 ...
 - Overloaded的方法是否可以改变返回值的类型
			
摘要: 重载Overload表示同一个类中可以有多个名称相同的方法,但这些方法的参数列表各不相同(即参数个数或类型不同) Overload是重载的意思,Override是覆盖的意思,也就是重写. 重载 ...
 - ZROI2018提高day6t2
			
传送门 分析 将所有字母分别转化为1~26,之后将字符串的空位补全为0,?设为-1,我们设dp[p][c][le][ri]表示考虑le到ri个字符串且从第p位开始考虑,这一位最小填c的方案数,具体转移 ...
 - 10.model/view实例(4)
			
任务:给表单的每一列添加列名. 思考: 1.只需要添加一个函数 headerData(). 横向方面添加列名 代码如下: QVariant MyModel::headerData(int sectio ...
 - Swing窗口Linux下不支持最大化问题
			
Swing窗口Linux下不支持最大化问题 摘自:https://www.linuxidc.com/Linux/2009-06/20519.htm [日期:2009-06-17] 来源:www.qua ...
 - window.location和window.location.href和document.location的关系
			
1,首先来区分window.location和window.location.href. window.location.href是一个字符串. 而window.location是一个对象,包含属性有 ...