转换地图 (康托展开+预处理+BFS)
Problem Description
1 2 3 4
8 7 6 5
对于这地图有三种具体操作方式:
A: 上下两行互换,如上图可变换为状态87654321
B: 每行同时循环右移一格,如上图可变换为41236785
C: 中间4个方块顺时针旋转一格,如上图可变换为17245368
根据所给的初状地图和终态地图,请给出完成转化的最少的变换步骤,若有多种变换方案则取字典序最小的那种。
Input
Output
SampleInput
12345678
17245368
12345678
82754631
SampleOutput
C
AC 最开始预处理了一下直接搜,本地爆炸。
然后想到了用康托展开打个表,然后。。。就AC了。
这里讲一下康托展开算法
X = An * (n-1)! + An-1 * (n-2)! + ... + A1 * 0!;
康拓展开就是求一个数字字符串在其全排列中的位置。
例如231这个数,全排列为123 132 213 231 312 321
所以231排在第4位,那么康托展开算法是如何求的呢。
例如求231的康托展开,从头至尾依次判断:
- 第一位数2,比2小的有一个,有1*2!
- 第二位数3,比3小的有两个,但2已经出现,有1*1!
- 第三位数1,有0*0!
累加起来就是2 + 1 = 3,表示比231小的有3个,所以231排在第4位。
代码实现的话就是:
int fac[] = {,,,,,,,,}; //i的阶乘
int kangtuo(int n,char a[]){ //n表示1~n个数,a数组表示数字
int i,j,t,res = ;
for(i = ; i < n; i++){
t = ;
for(j = i+; j < n; j++)
if(a[i] > a[j])
t++;
res += t*fac[n-i-];
}
return sum + ;
}
知道了康托展开后,就可以打表做了,值得一提的是这道题的预处理。因为题目输入两组字符串分别表示初始状态和结束状态,而我们打表是从12345678到各个状态的值,所以预处理我们把输入的初状态转成12345678,末状态也执行相应转换就可以了;
代码:
#include <iostream>
#include <string>
#include <cstdio>
#include <cstdlib>
#include <sstream>
#include <iomanip>
#include <map>
#include <stack>
#include <deque>
#include <queue>
#include <vector>
#include <set>
#include <list>
#include <cstring>
#include <cctype>
#include <algorithm>
#include <iterator>
#include <cmath>
#include <bitset>
#include <ctime>
#include <fstream>
#include <limits.h>
#include <numeric> using namespace std; #define F first
#define S second
#define mian main
#define ture true #define MAXN 1000000+5
#define MOD 1000000007
#define PI (acos(-1.0))
#define EPS 1e-6
#define MMT(s) memset(s, 0, sizeof s)
typedef unsigned long long ull;
typedef long long ll;
typedef double db;
typedef long double ldb;
typedef stringstream sstm;
const int INF = 0x3f3f3f3f; struct node{
string str,step;
}; bool vis[+];
int pos[],fac[] = {,,,,,,,,};
string ans[]; int fun(string a){
int i,j,t,sum = ;
for(i = ; i < ; ++i){
t = ;
for(j = i+; j < ; ++j)
if(a[i] > a[j])
++t;
sum += t*fac[-i-];
}
return sum+;
} void ma(string &s){
for(int i = ; i < ; ++i)
swap(s[i],s[i+]);
} string mb(string s){
string temp = s;
for(int i = ; i < ; ++i){
if(i== || i==)
temp[i]=s[i+];
else
temp[i]=s[i-];
}
return temp;
} void mc(string &s){
swap(s[],s[]);
swap(s[],s[]);
swap(s[],s[]);
} void bfs( string s ){
MMT(vis);
queue<node>q;
node pre,nxt; pre.str = s;
pre.step = "";
vis[fun(s)] = ;
ans[fun(s)] = pre.step;
q.push(pre); while(!q.empty()){
pre = q.front();
q.pop(); nxt = pre;
ma(nxt.str);
if(!vis[fun(nxt.str)]){
nxt.step += "A";
vis[fun(nxt.str)] = ;
ans[fun(nxt.str)] = nxt.step;
q.push(nxt);
} nxt.str = mb(pre.str);
if(!vis[fun(nxt.str)]){
nxt.step = pre.step + "B";
vis[fun(nxt.str)] = ;
ans[fun(nxt.str)] = nxt.step;
q.push(nxt);
} nxt = pre;
mc(nxt.str);
if(!vis[fun(nxt.str)]){
nxt.step += "C";
vis[fun(nxt.str)] = ;
ans[fun(nxt.str)] = nxt.step;
q.push(nxt);
}
}
} int main(){
ios_base::sync_with_stdio(false);
cout.tie();
cin.tie();
string s1,s2;
int k;
bfs("");
//12345678
//17245368
//12345678
//
while(cin>>s1>>s2){
swap(s1[],s1[]);
swap(s1[],s1[]);
swap(s2[],s2[]);
swap(s2[],s2[]);
for(int i = ; i < ; i++)
pos[s1[i]-''] = i+;
for(int i = ; i < ; i++)
s2[i] = pos[s2[i]-''];
k = fun(s2);
cout<<ans[k]<<endl;
}
return ;
}
其实康托展开也可以求逆运算,具体思想以及代码实现这里就不讲了=7=
转换地图 (康托展开+预处理+BFS)的更多相关文章
- UESTC 485 Game(康托展开,bfs打表)
Game Time Limit: 4000/2000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Submit Status t ...
- 康托展开+反向bfs
康托展开+反向bfs hdu 1043 题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1043 #include <iostream> # ...
- ACM-康托展开+预处理BFS之魔板——hdu1430
魔板 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submis ...
- HDU1430 BFS + 打表 + 康托展开
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1430 , 一道比较好的题. 这道题要用到很多知识,康托展开.BFS.打表的预处理还要用到一一映射,做完 ...
- hdu 1430 (BFS 康托展开 或 map )
第一眼看到这题就直接BFS爆搜,第一发爆了内存,傻逼了忘标记了,然后就改,咋标记呢. 然后想到用map函数,就8!个不同的排列,换成字符串用map标记.然后又交一发果断超时,伤心,最恨超时,还不如来个 ...
- HDU 1430 魔板(康托展开+BFS+预处理)
魔板 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submis ...
- HDU 1043 & POJ 1077 Eight(康托展开+BFS+预处理)
Eight Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 30176 Accepted: 13119 Special ...
- hdu3567 八数码2(康托展开+多次bfs+预处理)
Eight II Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 130000/65536 K (Java/Others)Total S ...
- hdu1430魔板(BFS+康托展开)
做这题先看:http://blog.csdn.net/u010372095/article/details/9904497 Problem Description 在魔方风靡全球之后不久,Rubik先 ...
随机推荐
- (十八)c#Winform自定义控件-提示框
前提 入行已经7,8年了,一直想做一套漂亮点的自定义控件,于是就有了本系列文章. 开源地址:https://gitee.com/kwwwvagaa/net_winform_custom_control ...
- c# 将dwg文件转化为pdf
https://blog.csdn.net/mywaster/article/details/50220379 最近做一个项目,要求将dwg文件转化为pdf,开发工具VS2010 + AutoCad ...
- UI 组件 | Button
最近在与其他自学 Cocos Creator 的小伙伴们交流过程中,发现许多小伙伴对基础组件的应用并不是特别了解,自己在编写游戏的过程中也经常对某个属性或者方法的用法所困扰,而网上也没有比较清晰的用法 ...
- (18)ASP.NET Core 基于现有数据库创建EF模型(反向工程)
1.简介 Entity Framework Core可通过数据库提供给应用程序的插件访问许多不同的数据库.我们可以通过使用Entity Framework Core构建执行基本数据访问的ASP.NET ...
- Sqlserver将表中某列数据以符号分成多行
WITH testtb2 AS ( UNION ALL ) ) ) ) ) PERCENT SUBSTRING(VisitorCard, STA - LENS, LENS) AS OrderReque ...
- peewee
字段查看http://docs.peewee-orm.com/en/latest/peewee/models.html#fields 方法使用https://blog.csdn.net/qq_3962 ...
- 记录一次Jquery中 this 关键字使用出现的问题
今天在用Jquery改造之前的JS代码过程中,遇到了一个让我懵逼了三小时的问题. 问题的关键在 this 的使用.在这里与大家分享一下.并且分享一下我做表单提交的检查代码 错误代码如下: $(&quo ...
- Dubbo(一):Dubbo运行原理
前言: 在开始入门Javaweb时,学的基本都是MVC开发模式,一个项目基本上就是model,view,controller三层.但是随着系统的服务逐渐加多,SOA模式更加适合目前项目开发.而SOA模 ...
- MinorGC 和 FullGC的理解
1.GC回收机制熟悉么,分代算法知道么?2.了解 Java 虚拟机的垃圾回收算法? 从年轻代空间(包括 Eden 和 Survivor 区域)回收内存被称为 Minor GC. Major GC 是清 ...
- from 表单用 GET 方法进行 URL 传值时后台无法获取问题
问题描述 <a href="${pageContext.request.contextPath}/client?method=add">点我</a> < ...