【LOJ3099】[SNOI2019]积木(搜索)
lca 学长出的我省省选的神仙题目 省强我菜系列
题目
分析
我可能说不清楚,对着代码理解吧 …… 感觉这题的主要难点是:不要想他具体是怎么操作的,只要知道他一定存在一种操作方式能够实现就行了。
首先要注意到一个很重要的性质:对于当前空格所在的点,除非这个点在目标中就是空格,否则一定可以通过一步操作使这个点变得和目标中一样(下称「变得和目标中一样」为「归位」)。例如,如果这个点最终是朝上的(意思是这个点是一个上下方向的块的下半部分,朝下、朝左、朝右同理),那么无论它上面是一个朝上还是朝左还是朝右(显然不可能朝下)的块,都可以通过移动它上面的块来使这个点变得朝上。这样只需要不断地通过这种操作使空格所在的块归位,就可以让空格到达目标中的空格的位置。
但是,这个过程中不能保证访问到了所有格子,所以有些未归位的格子可能没有被发现。因此在把空格归位后要去主动寻找未归位的格子。具体方法是用 dfs 主动寻找未归位的格子。虽然这样会让已经归位的格子变成未归位,但是回溯的时候挪回去就行了。每找到一个未归位的格子就像上一段描述的一样使它归位,直到空格又回到 dfs 找到的这个地方。如此把所有格子都 dfs 一遍,就能让所有格子归位。
代码
#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;
namespace zyt
{
const int N = 2e3 + 10, M = 8e6 + 10, CH = 128;
const int dx[] = {0, 0, 1, -1}, dy[] = {1, -1, 0, 0};
const char opt[] = {'R', 'L', 'D', 'U'}, ch[] = {'<', '>', 'n', 'u'};
char a[N][N], b[N][N], ans[M];
bool vis[N][N];
int cntans, n, m, dir[CH];
void move(const int d, int &x, int &y)
{
ans[cntans++] = opt[d];
int x1 = x + dx[d], y1 = y + dy[d];
if (x1 < 0 || x1 >= n || y1 < 0 || y1 >= m)
fprintf(stderr, "gg");
int x2 = x1 + dx[dir[a[x1][y1]]], y2 = y1 + dy[dir[a[x1][y1]]];
a[x][y] = ch[d], a[x1][y1] = ch[d ^ 1], a[x2][y2] = 'o';
x = x2, y = y2;
}
void go(int &x, int &y, const int x1, const int y1)
{
while (x != x1 || y != y1)
move(dir[b[x][y]], x, y);
}
void dfs(int x, int y)
{
vis[x][y] = true;
for (int d = 0; d < 4; d++)
{
int x1 = x + dx[d], y1 = y + dy[d];
if (x1 < 0 || x1 >= n || y1 < 0 || y1 >= m || vis[x1][y1])
continue;
if (a[x1][y1] != b[x1][y1])
{
int x0 = x, y0 = y;
move(d, x, y);
go(x, y, x0, y0);
}
vis[x1][y1] = true;
int dd = dir[a[x1][y1]];
move(d, x, y);
dfs(x, y);
move(dd ^ 1, x, y);
}
}
int work()
{
dir['<'] = 0, dir['>'] = 1, dir['n'] = 2, dir['u'] = 3;
scanf("%d%d", &n, &m);
for (int i = 0; i < n; i++)
scanf("%s", a[i]);
for (int i = 0; i < n; i++)
scanf("%s", b[i]);
int x, y, x1, y1;
for (int i = 0; i < n; i++)
for (int j = 0; j < m; j++)
{
if (a[i][j] == 'o')
x = i, y = j;
if (b[i][j] == 'o')
x1 = i, y1 = j;
}
go(x, y, x1, y1);
dfs(x, y);
printf("%s", ans);
return 0;
}
}
int main()
{
return zyt::work();
}
【LOJ3099】[SNOI2019]积木(搜索)的更多相关文章
- NOIP 模拟 玩积木 - 迭代加深搜索 / bfs+hash+玄学剪枝
题目大意: 有一堆积木,0号节点每次可以和其上方,下方,左上,右下的其中一个交换,问至少需要多少次达到目标状态,若步数超过20,输出too difficult 目标状态: 0 1 1 2 2 2 3 ...
- 「SNOI2019」积木
传送门 Description 有一块\(n\)行\(m\)列的网格板, \(n,m\)都是奇数.网格上平铺着一些\(1*2\)的积木.积木可以旋转,不能重叠.网格板上只有一格的空位. 你可以做两种操 ...
- 洛谷P2409 Y的积木
P2409 Y的积木 77通过 491提交 题目提供者zhouyonglong 标签云端评测 难度普及+/提高 提交 讨论 题解 最新讨论 这组数据几乎可以卡掉所有程- 第一个题解有点问题 求教大 ...
- 深度优先搜索(DFS)递归形式改为非递归形式
DFS将递归改为非递归这个方法的需求来自于一道三维积木组合的题目,还在苦苦调试中,暂且不提. 普通的认识对于递归向非递归的转化无非是使用栈,但是结合到深度搜索如何将栈很好利用,如何很好保存现场,都不是 ...
- XJOI1657&Codevs1255搭积木【树状动规】
搭积木 一种积木搭建方式,高为H的积木,最底层有M个积木,每一层的积木数是他的低一层的积木数+1或-1.总共有N个积木.(且每行积木数不超过10)比如上图N=13 H=6 M=2. 输入格式: 第一行 ...
- C# 全文搜索Lucene
全文出自:https://blog.csdn.net/huangwenhua5000/article/details/9341751 1 lucene简介1.1 什么是luceneLucene是一个全 ...
- code1225 搭积木
题目分析:将当前层定义为第h层,共用了n块积木,本层积木数为m,f(h,n,m) 那么可以扩展数两种状态:f(h-1,n-m,m-1),f(h-1,n-m,m+1) 直接搜索可能的数据达到h^m,超时 ...
- codevs 1255 搭积木 x
1255 搭积木 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 大师 Master 题目描述 Description 一种积木搭建方式,高为H的积木,最底层有M个积木,每一层的积木 ...
- 基于层级表达的高效网络搜索方法 | ICLR 2018
论文基于层级表达提出高效的进化算法来进行神经网络结构搜索,通过层层堆叠来构建强大的卷积结构.论文的搜索方法简单,从实验结果看来,达到很不错的准确率,值得学习 来源:[晓飞的算法工程笔记] 公众号 ...
随机推荐
- 18-Flutter移动电商实战-首页_火爆专区商品接口制作
1.获取接口的方法 在service/service_method.dart里制作方法.我们先不接收参数,先把接口调通. Future getHomePageBeloConten() async{ ...
- UFUN函数 UF_ASSEM UF_PART函数(UF_ASSEM_ask_work_part,UF_PART_ask_part_name)
UF_initialize(); tag_t work_part_tag=NULL_TAG; ]=""; //获取当前工作部件的tag work_part_tag=UF_ASSEM ...
- 洛谷 P5269 欧稳欧再次学车 题解
P5269 欧稳欧再次学车 题目背景 请自行脑补一张欧稳欧学车的图 题目描述 欧稳欧学车时经常用一辆橡树车练习.这辆橡树车共有 \(N\) 个挡位,欧稳欧每秒可以把挡位增加或减少 \(1\),初始时( ...
- uniapp登录流程详解uni.login
uni.login(OBJECT)登录 H5平台登陆注意事项: 微信内嵌浏览器运行H5版时,可通过js sdk实现微信登陆,需要引入一个单独的js,详见普通浏览器上实现微信登陆,并非开放API,需要向 ...
- js 队列
js 中的异步队列(micro & macro) js都是靠事件驱动的, js中的事件循环机制是什么呢? 只是简单写一下自己的理解, 所以不是很全面; js 程序执行有 主队列 以及 异步队列 ...
- Google Dremel架构
Dremel 是Google 的“交互式”数据分析系统.Google开发了Dremel将处理时间缩短到秒级,作为MapReduce的有力补充.Apache推出Dremel的开源实现Drill,将Dre ...
- unity EditorGUILayer绘制报错
最近在开发一个可视化工具的时候,遇到了一个代码错误,小小的记录一下 具体的报错信息:ArgumentException: Getting control 0's position in a group ...
- linux安装yasm报错
进入yasm-1.2.0, 输入指令 ./configure //编译yasm make && make install //安装yasm,安装完成即可. 报错信息 make[2]: ...
- MySQL索引失效的几种场景
我们都知道建立索引能够提高查询效率,那么是不是任何情况下都能提高呢,当然不是的的,下面我们就来列举一些常见的索引失效的场景. 借用上一篇文章的dm_person_info表 在card_code列没加 ...
- Python 使用scapy 时报:ImportError: cannot import name 'NPCAP_PATH' 解决
解决办法:下载源码 https://github.com/secdev/scapy 下载完成后解压后里面有个scapy文件夹,把这个文件夹替换\Lib\site-packages\ 下的scapy 整 ...