转换地图 (康托展开+预处理+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先 ...
随机推荐
- 《机器学习技法》---线性SVM
(本文内容和图片来自林轩田老师<机器学习技法>) 1. 线性SVM的推导 1.1 形象理解为什么要使用间隔最大化 容忍更多的测量误差,更加的robust.间隔越大,噪声容忍度越大: 1.2 ...
- 依赖注入在 dotnet core 中实现与使用:1 基本概念
关于 Microsoft Extension: DependencyInjection 的介绍已经很多,但是多数偏重于实现原理和一些特定的实现场景.作为 dotnet core 的核心基石,这里准备全 ...
- 【Node.js】 bodyparser实现原理解析
为什么我们需要body-parser 也许你第一次和bodyparser相遇是在使用Koa框架的时候.当我们尝试从一个浏览器发来的POST请求中取得请求报文实体的时候,这个时候,我们想,这个从Koa自 ...
- 微信小程序项目总结-记账小程序(包括后端)
一.小程序部分 这是理财系统的前端,江苏海洋大学微信小程序比赛,最后获得了一等奖 GitHub:https://github.com/GeorgeLeoo/finance 1. 项目描述 (1). 此 ...
- 小白学Python(8)——pyecharts 入门
简介: pyecharts 是一个用于生成 Echarts 图表的类库. echarts 是百度开源的一个数据可视化 JS 库,主要用于数据可视化.pyecharts 是一个用于生成 Echarts ...
- C#表达式树浅析
一.前言 在我们日常开发中Lamba 表达式经常会使用,如List.Where(n=>Name="abc") 使用起来非常的方便,代码也很简洁,总之一个字就是“爽”.在之前我 ...
- 重读《学习JavaScript数据结构与算法-第三版》- 第6章 链表(一)
定场诗 伤情最是晚凉天,憔悴厮人不堪言: 邀酒摧肠三杯醉.寻香惊梦五更寒. 钗头凤斜卿有泪,荼蘼花了我无缘: 小楼寂寞新雨月.也难如钩也难圆. 前言 本章为重读<学习JavaScript数据结构 ...
- redux 源码阅读
目录 [目录结构] [utils] actionTypes.js isPlainObject.js warning.js [逻辑代码] index.js createStore.js compose. ...
- 深入剖析PHP7内核源码(二)- PHP变量容器
简介 PHP的变量使用起来非常方便,其基本结构是底层实现的zval,PHP7采用了全新的zval,由此带来了非常大的性能提升,本文重点分析PHP7的zval的改变. PHP5时代的ZVAL typed ...
- 2. Sentinel源码分析—Sentinel是如何进行流量统计的?
这一篇我还是继续上一篇没有讲完的内容,先上一个例子: private static final int threadCount = 100; public static void main(Strin ...