题目描述:

代码如下:

 #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. python日常小计

    1.查看变量类型:  pring type(item) 2.解决list中的中文显示乱码 使用decode('string_escap')将数据库查询返回的将带转义的字节码字符串转换为成utf-8中文

  2. 关于SpringCloud配置网关转发时出现一下啊错误:“com.netflix.zuul.exception.ZuulException: Forwarding error at org.springframework.cloud.netflix.zuul.filters.route.RibbonRoutingFilter.handleException”

    com.netflix.zuul.exception.ZuulException: Forwarding error at org.springframework.cloud.netflix.zuul ...

  3. Goland配置

    Global GOPATH 用来设置所有go项目的大目录 Project GOPATH 用来设置单项目目录 2个目录必须配置

  4. Spring AOP选择切点的问题

    先上代码: /** * 管理员登录方法的切入点 */ @Pointcut("execution(* com.arch.shiro.realm.UserRealm.doGetAuthentic ...

  5. flex 布局 实现电商页面商品展示floor

    有了上一篇,对flex的初次使用,心里痒痒的试着,实现了电商首页,商品展示的floor,先看下效果: 要实现首先是对组件构件的拆解,拆解如下: 页面布局如下: <div class=" ...

  6. plsql插入数据出现乱码问题

    今天在使用plsql 插数据时 ,遇到的问题就是插入进去是乱码问题,在这里总结一下如何解决的 首先声明一下就是: 现在只有客户端,没有服务端,有一些需求就是总是有许多数据要插入进去,但是在使用sql语 ...

  7. React native中使用XMLHttpRequest请求数据

    一.代码 import React, { Component } from 'react'; import { AppRegistry, StyleSheet, Text, View } from ' ...

  8. Java中语法与C/CPP的区别

    static不能在成员方法中定义,只能作为类变量定义.

  9. apply,call以及bind的区别

    每个函数都包含两个非继承而来的方法:apply()和 call(). 这两个方法的用途都是在特定的作用域中调用函数,实际上等于设置函数体内 this 对象的值. 一.apply() apply()方法 ...

  10. Mawawa CSS 学习之旅 Display

    CSS 类型之 Display 更新时间: 2018-2-10: 一个良好的布局结构从 display 开始! 分类:外部值.内部值.列表值.属性值.混合值.显示值.全局值: 一.外部值 作用:主要用 ...