首页 > 其他分享 >P1126 机器人搬重物 题解

P1126 机器人搬重物 题解

时间:2023-10-01 20:23:11浏览次数:48  
标签:oj oi int 题解 重物 vis tk && P1126

Problem

题目概括

  • $n \times m $ 的网格,有些格子是障碍格。\(0\) 无障碍,\(1\) 有障碍。机器人有体积,总是在格点上

  • 有5种操作:

    • 向前移动 \(1/2/3\) 步
    • 左转 \(/\) 右转
  • 每次操作需要 \(1\) 秒。

  • 求从 \(x_1,y_1\) 到 \(x_2,y_2\) 点的最短路。

  • 机器人有一个初始方向 $ E / S / W / N$ (东南西北)。

思路

  1. 考察算法:BFS求最短路。
  2. 实现过程 \(\&\) 注意点:
      1. 障碍是在格子上的,需要将周围 \(4\) 个点标记为障碍。
      1. 本题下标从 \(0\) 开始
      1. 由于机器人有体积,所以可走的范围只有 \((1,1) ~ (n - 1,m - 1)\) !
      1. 在一次性走 \(2,3\) 步的时候,一定要判断 \(a\) 数组中路上的每一个点是否为 \(0\)。不然会出现 \(“\) 穿墙 \(”\) 的情况。

定义一个 queue<node> q; ,然后 \(node\) 里面有 \(4\) 个信息:\(x,y,k,step\)。 \(k\) 代表当前的方向在方向数组中的下标,\(step\) 表示花了几秒 \((\)走了几步\()\)。

然后本题 \(bool\) 类型的 \(vis\) 数组可以开成三维的。
\(vis[x][y][k]\) 代表 \(x,y\) 点的 \(k\) 方向已经被走过了。

然后 \(BFS\) ,就可以 \(AC\) 了~

代码

#include <bits/stdc++.h>

using namespace std;

const int N = 60;
struct node {
    int X, Y, K, stp;
};
queue<node> q;
// 方向数组
int fx[5] = {0, 1, 0, -1};
int fy[5] = {1, 0, -1, 0};
int n, m;
int a[N][N], sx, sy, ex, ey, k, xx;
bool vis[N][N][5];
char c;
map<char, int> mp; // 映射四个方向对应方向数组的下标(0-3)

// 将障碍格四周的点标记为1
void fun(int x, int y) {
  a[x][y] = a[x - 1][y - 1] = a[x - 1][y] = a[x][y - 1] = 1;
}

// 检验坐标是否合法
bool check(int x, int y) { return x > 0 && x < n && y > 0 && y < m; }

// 左转
int TurnLeft(int tmpk) { return (tmpk + 3) % 4; }
// 右转
int TurnRight(int tmpk) { return (tmpk + 5) % 4; }

int bfs() {
    // 初始化
    q.push((node) {sx, sy, k, 0});
    vis[sx][sy][k] = true;
    while (!q.empty()) {
        int x = q.front().X, y = q.front().Y, tk = q.front().K, step = q.front().stp;
        q.pop();
        // 到达终点
        if (x == ex && y == ey) return step;
        // 左转 & 右转
        if (!vis[x][y][TurnLeft(tk)] && !a[x][y]) {
            q.push((node) {x, y, TurnLeft(tk), step + 1});
            vis[x][y][TurnLeft(tk)] = true;
        }
        if (!vis[x][y][TurnRight(tk)] && !a[x][y]) {
            q.push((node){x, y, TurnRight(tk), step + 1});
            vis[x][y][TurnRight(tk)] = true;
        }
        int oi = x + fx[tk], oj = y + fy[tk];
        int oi_2 = x + 2 * fx[tk], oj_2 = y + 2 * fy[tk];
        int oi_3 = x + 3 * fx[tk], oj_3 = y + 3 * fy[tk];
        // 走 1/2/3 步
        if (!vis[oi][oj][tk] && !a[oi][oj] && check(oi, oj)) {
            q.push((node){oi, oj, tk, step + 1});
            vis[oi][oj][tk] = true;
        }
        if (check(oi_2, oj_2) && !vis[oi_2][oj_2][tk] && !a[oi][oj] && !a[oi_2][oj_2]) {
            q.push((node){oi_2, oj_2, tk, step + 1});
            vis[oi_2][oj_2][tk] = true;
        }
        if (check(oi_3, oj_3) && !vis[oi_3][oj_3][tk] && !a[oi][oj] && !a[oi_2][oj_2] && !a[oi_3][oj_3]) {
            q.push((node){oi_3, oj_3, tk, step + 1});
            vis[oi_3][oj_3][tk] = true;
        }
    }
    return -1; // 走不到
}


int main() {
    mp['E'] = 0, mp['S'] = 1, mp['W'] = 2, mp['N'] = 3; // 初始化方向
    scanf("%d%d", &n, &m);
    for (int i = 1; i <= n; i++) {
        a[i][0] = a[0][i] = a[i][m] = a[n][i] = 1;
        for (int j = 1; j <= m; j++) {
            scanf("%d", &xx);
            if (xx == 1) fun(i, j); // 障碍点四周的点都标记为1
        }
    }
    scanf("%d%d%d%d", &sx, &sy, &ex, &ey);
    cin >> c;
    k = mp[c]; // 映射方向
    printf("%d", bfs());
    return 0;
}

标签:oj,oi,int,题解,重物,vis,tk,&&,P1126
From: https://www.cnblogs.com/yhx0322/p/17739216.html

相关文章

  • P1182 数列分段 Section II 题解
    Problem考察知识点:二分、贪心。题目描述对于给定的一个数组,现要将其分成\(M\)段,并要求每段连续,且每段和的最大值最小。思路二分答案出每段和最大值的最小值,然后贪心检验是否满足。难点在\(check\)上。策略:每次开始循环,如果没有超范围,就一直选,知道选满为止,求最大值。代......
  • Codeforces 1278D 题解
    题目大意题目大意给你\(n\)(\(1\leqslantn\leqslant5\cdot10^5\))条线段\([l_1,r_1],[l_2,r_2],\cdots,[l_n,r_n]\)(\(1\lel_i<r_i\le2n\))。保证每条线段的端点为整数,且\(\foralli,j\)(\(i\nej\)),不存在\(l_i=l_j\)或\(r_i=r_j\),不存......
  • P5943 [POI2002] 最大的园地 题解
    题目传送门前置知识单调栈简化题意在一个\(n\timesn\)的正方形内找到最大的由\(0\)组成的子矩形的面积。解法令\(f_{i,j}(1\lei,j\len)\)表示从\((1,j)\)到\((i,j)\)中以\((i,j)\)结尾的均为\(0\)的子串长度,即\((i,j)\)上面可延伸的最大距离(子矩形的......
  • Codeforces 1702G2 题解
    题目大意给出一个大小为\(n\)的树,\(q\)次询问,每次给出一个大小为\(m\)的点集,判断是否有一条链覆盖这些点(这条链可以经过其他点)。\(n,\summ\leqslant2\cdot10^5\),\(q\leqslant10^5\)。提示提示1思考将$m$个点按深度排序。题解题解考虑将\(m\)个点按树......
  • ABC322G题解
    这场的G怎么这么毒瘤啊/kk听说正解是DP?我爆搜头一个表示不服!statement找出三元组\((S,a,b)\)的数量,使得\(S\)在\(a\)进制下和在\(b\)进制下的差为\(X\),其中\(0\leqS_i<(min(a,b,10))\)。首先因为\(X>0\)显然\(S\)不可能为\(1\)位数。如果\(S\)......
  • 「题解」Codeforces Round 895 (Div. 3)
    A.TwoVesselsProblem题目Sol&Code签到题#include<bits/stdc++.h>typedeflonglongll;intmin(inta,intb){returna<b?a:b;}intmax(inta,intb){returna>b?a:b;}intT,a,b,c;intmain(){scanf("%d"......
  • AT_abc254_h [ABC254Ex] Multiply or Divide by 2 题解
    打篇题解巩固一下。题意给你两个集合\(A\)和\(B\),对于任意\(A\)集合中的元素,我们可以进行\(2\)种操作:\(a_i\gets\left\lfloor\frac{a_i}{2}\right\rfloor\)或\(a_i\gets2\timesa_i\)。问最少要多少次才能使\(A\)和\(B\)中的元素相等。分析首先我们可以令\(a......
  • UVA12655 Trucks 题解
    题目传送门前言中文题目可以看link。前置知识Kruskal重构树|最近公共祖先简化题意给定一个\(N\)个点\(M\)条边的有向图,共有\(S\)次询问,每次询问从\(L\)到\(H\)所有的路径中最小的权值的最大值(多组数据)。本题即最大瓶颈路问题。解法使最小的权值最大,不难......
  • 【洛谷 P1305】新二叉树 题解(结构体数组+先序遍历+二叉树)
    新二叉树题目描述输入一串二叉树,输出其前序遍历。输入格式第一行为二叉树的节点数。()后面行,每一个字母为节点,后两个字母分别为其左右儿子。特别地,数据保证第一行读入的节点必为根节点。空节点用*表示输出格式二叉树的前序遍历。样例#1样例输入#16abcbdicj*d**i**j**......
  • SP16113 SUBTLEBA - Trucks Transportation 题解
    题目传送门前言本题样例有问题,如果想要样例可以去vjudge上。本题提交后可能会出现UKE,建议前往link提交,而且本篇题解中所提供的代码也为link代码。前置知识Kruskal重构树|最近公共祖先简化题意给定一个\(N\)个点\(M\)条边的有向图,共有\(S\)次询问,每次询问......