题目描述:

代码如下:

 #include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define N 1000000
#define HN 1000003
#define LEN 9 int head[HN],next[N];
int st[N][LEN], goal[LEN];
int dis[N]; //记录步数 int Hash(int *st)//获取本次排列组合的哈希值
{
int i,v;
v = ;
for (i= ; i<LEN ; i++)//得到一个LEN长度数值:v
{
v = v* + st[i];
}
return v%HN;//得到对应哈希值
} // 功能:对本次的排列组合,若不在哈希表中,则在哈希表中进行键值映射
int try_insert(int rear)//rear:本次组合的编号
{
int h = Hash(st[rear]); //获得本次排列组合对应的哈希值
int u = head[h]; //根据哈希值,获得哈希表中对应的键(key),若没有则为0 while (u)//当上一步有拿到键(key),查找对应的值(value->排列组合),
{
if (memcmp(st[u],st[rear],sizeof(st[]))==)//存在相同的值(value->排列组合)
return ; //本次值(value->排列组合)将不记录,退出
u = next[u]; //当前键(key)还有其他值,遍历获得其他值(value->排列组合)
} //当前值(value->排列组合)不在哈希表中,将进行记录
next[rear] = head[h]; //当前键(key)的记录是否有其他值,若本身为新组合,为0;若为存在组合,则为其上一个编号(相同组合)
head[h] = rear; //哈希表中对应的键(key),更新为本次组合编号(rear)
return ;
} int bfs()
{
int d;
int x,y,z,nx,ny,nz;
int fron = ,rear = ; //fron:当前排列组合;rear:移动后的排列组合
const int dir[][] = {{,},{,-},{,},{-,},}; memset(head,,sizeof(head));
while (fron < rear)
{
if (memcmp(goal,st[fron],sizeof(st[]))==)//当前排列组合为目标排列组合
return fron; for (z= ; z<LEN ; z++)
{
if (st[fron][z]==)//0即为对应的'.'
break;
} x = z/,y = z%;//获得'.'的坐标
for (d= ; d< ; d++)
{
nx = x+dir[d][];
ny = y+dir[d][];
if (nx>= && nx< && ny>= && ny<)
{
nz = *nx+ny;//'.'将要移动的下一步位置 memcpy(&st[rear],&st[fron],sizeof(st[]));
st[rear][nz] = st[fron][z];
st[rear][z] = st[fron][nz]; dis[rear] = dis[fron]+;//记录步数
if (try_insert(rear))//对新排列组合进行查找
rear ++;//进队列
}
}
fron ++;//出队列
} return ;
} int main(void)
{
int i,ans;
char s1[LEN+],s2[LEN+];
memset(dis,,sizeof(dis));
scanf("%s%s",&s1,&s2); for (i= ; i<LEN ; i++)//将'.'转换为0,便于计算哈希值
{
if (s1[i]=='.')
st[][i] = ;
else
st[][i] = s1[i]-''; if (s2[i]=='.')
goal[i] = ;
else
goal[i] = s2[i]-'';
} ans = bfs();
if (ans > )
printf("%d",dis[ans]);
else
puts("-1"); return ;
}

C解法

解题思路:

本题要求最少步数转换成目标组合,广度优先遍历(BFS)可满足要求

每次对新组合进行检查时,需要较高的检索效率,

1.使用map或者set做检索时,其对应的检索效率低,超时不能满足题目需求;

2.对组合进行哈希判重,大大提高了检索效率;

关于本题的哈希判重,参考了:

https://blog.csdn.net/hao_zong_yin/article/details/62419919

https://www.cnblogs.com/acxblog/p/7253477.html

哈希判重,利用键-值映射,有效提高了搜索的效率;

1.建立键(key)表,head[NH],需初始化;建立值(value)表,next[N];

2.BFS搜索新组合,通过向不同方向移动 ' . ' (0)  得到新的组合;

3.根据新组合,计算对应的哈希值;

4.根据哈希值,进行检索决定是否添加新的排列组合

  4.1.插入的组合存在

  4.2.插入的组合不存在

5.若有新组合添加,队列+1(rear+1)

6.每次出列(fron+1),检查是否为目标组合

7.若fron > rear,则不存在方案,输出-1

[蓝桥杯]PREV-19.历届试题_九宫重排的更多相关文章

  1. 蓝桥杯练习系统历届试题 剪格子 dfs

    问题描述 如下图所示,3 x 3 的格子中填写了一些整数. +--*--+--+|10* 1|52|+--****--+|20|30* 1|*******--+| 1| 2| 3|+--+--+--+ ...

  2. 蓝桥杯练习系统历届试题 带分数 dfs

    问题描述 100 可以表示为带分数的形式:100 = 3 + 69258 / 714. 还可以表示为:100 = 82 + 3546 / 197. 注意特征:带分数中,数字1~9分别出现且只出现一次( ...

  3. [蓝桥杯]PREV-44.历届试题_青蛙跳杯子

    问题描述 X星球的流行宠物是青蛙,一般有两种颜色:白色和黑色. X星球的居民喜欢把它们放在一排茶杯里,这样可以观察它们跳来跳去. 如下图,有一排杯子,左边的一个是空着的,右边的杯子,每个里边有一只青蛙 ...

  4. [蓝桥杯]PREV-10.历届试题_幸运数

    问题描述 幸运数是波兰数学家乌拉姆命名的.它采用与生成素数类似的“筛法”生成 . 首先从1开始写出自然数1,,,,,,.... 就是第一个幸运数. 我们从2这个数开始.把所有序号能被2整除的项删除,变 ...

  5. [蓝桥杯]PREV-27.历届试题_蚂蚁感冒

    问题描述 长100厘米的细长直杆子上有n只蚂蚁.它们的头有的朝左,有的朝右. 每只蚂蚁都只能沿着杆子向前爬,速度是1厘米/秒. 当两只蚂蚁碰面时,它们会同时掉头往相反的方向爬行. 这些蚂蚁中,有1只蚂 ...

  6. [蓝桥杯]PREV-26.历届试题_最大子阵

    问题描述 给定一个n*m的矩阵A,求A中的一个非空子矩阵,使这个子矩阵中的元素和最大. 其中,A的子矩阵指在A中行和列均连续的一块. 输入格式 输入的第一行包含两个整数n, m,分别表示矩阵A的行数和 ...

  7. [蓝桥杯]PREV-25.历届试题_城市建设

    问题描述 栋栋居住在一个繁华的C市中,然而,这个城市的道路大都年久失修.市长准备重新修一些路以方便市民,于是找到了栋栋,希望栋栋能帮助他. C市中有n个比较重要的地点,市长希望这些地点重点被考虑.现在 ...

  8. [蓝桥杯]PREV-23.历届试题_数字游戏

    问题描述 栋栋正在和同学们玩一个数字游戏. 游戏的规则是这样的:栋栋和同学们一共n个人围坐在一圈.栋栋首先说出数字1.接下来,坐在栋栋左手边的同学要说下一个数字2.再下面的一个同学要从上一个同学说的数 ...

  9. [蓝桥杯]PREV-22.历届试题_国王的烦恼

    问题描述 C国由n个小岛组成,为了方便小岛之间联络,C国在小岛间建立了m座大桥,每座大桥连接两座小岛.两个小岛间可能存在多座桥连接.然而,由于海水冲刷,有一些大桥面临着不能使用的危险. 如果两个小岛间 ...

随机推荐

  1. SmartBinding与kbmMW#1

    即将发布的kbmMW,实现了SmartBinding,SmartBinding的设计目标是: 必须易于使用 必须最小化或完全删除锅炉板代码.(你看到这里的趋势了吗?... kbmMW从那时开始就是为了 ...

  2. echarts绘制k线图为什么写candlestick类型就报错

    错误提示:echarts-en.common.js:11713 Uncaught Error: Component series.candlestick not exists. Load it fir ...

  3. Ubuntu软件中心闪退

    sudo apt-get update sudo apt-get dist-upgrade sudo apt-get install --reinstall software-center

  4. shell脚本登录远程服务器并下载至本地

    通常有这样备份的需求,将远程服务器的代码或者数据打包压缩然后下载到本地路径 实现方式 需要对远程服务器实现无密码访问,通过配置公钥实现: 使用ssh执行命令然后转向到本地的方法一步完成打包和下载,可参 ...

  5. ELK简单安装测试

    1 介绍组件 Filebeat是一个日志文件托运工具,在你的服务器上安装客户端后,filebeat会监控日志目录或者指定的日志文件,追踪读取这些文件(追踪文件的变化,不停的读). Kafka是一种高吞 ...

  6. jdk的环境变量配置

    首先下载jdk,下载地址:http://www.oracle.com/technetwork/java/javase/downloads/index.html 可以选择下载jdk的版本,按照提示一步一 ...

  7. Golang安装与环境的配置

    (一) 不需FQ的情况 打开https://studygolang.com/ 下载相应的系统安装包. (二) 安装与环境配置 linux: 解压到指定目录: sudo tar zxvf [安装包的下载 ...

  8. 锋利的jQuery初学(4)

    css选择器与jQuery选择器 css选择器 符号 说明 用法 #id 选择器 #id{} .class 类选择器 .class{} Element 标签选择器 p{} , 并集选择器 div,p{ ...

  9. ORA-28000: the account is locked解决

    首先使用具有sysdba权限的账户登陆,如sys账户和system账户 新建一个sql窗体,并执行语句解锁被锁定的账户,如我这里sgyw账户: alter user sgyw account unlo ...

  10. 使用JBolt新建Maven版工程步骤

    一.打开新建对话框 在左侧右键new中可以找到JFinal创建工程的菜单 JBoltHome页面也有快捷按钮用来弹出创建工程对话框. 二.填写Maven和其他信息配置 填写工程name 主包名 下面有 ...