11. 미로 탈출 명령어

aj4941·2023년 8월 31일
0

https://school.programmers.co.kr/learn/courses/30/lessons/150365#

문제를 접근하는 과정에서 dp[x][y][k] = true의 boolean dp로 접근하는 생각이 가장 먼저 들었고 이를 역추적하여 문자열을 얻어낼 수 있다 생각했다.
그런데 (x, y) -> (r, c)로 이동하면서 문자열을 얻어내야 사전순 최소를 구할 수 있었고 dp를 (r, c, k) -> (x, y, 0) 형태로 구해야 함을 알 수 있었다.
이렇게 역추적을 하는 방향에 따라서 구해야 하는 dp 값이 달라질 수 있음에 유의해야 한다.

#include <bits/stdc++.h>
using namespace std;
const int N = 52;
int dp[N][N][N * N];
int dx[4] = { 1, 0, 0, -1 };
int dy[4] = { 0, -1, 1, 0 };
char ch[4] = { 'd', 'l', 'r', 'u' };

void solve(int x, int y, int k, int n, int m)
{
    // cout << x << ' ' << y << ' ' << k << endl;
    int &ret = dp[x][y][k];
    if (ret != -1) return;
    ret = 1;
    if (k == 0) return;
    
    for (int i = 0; i < 4; i++)
    {
        int nx = x + dx[i], ny = y + dy[i];
        if (nx <= 0 || ny <= 0 || nx > n || ny > m) continue;
        solve(nx, ny, k - 1, n, m);
    }
}

string solution(int n, int m, int x, int y, int r, int c, int k) 
{
    memset(dp, -1, sizeof dp);
    solve(r, c, k, n, m);
    string ans = "";
    
    int sk = 0;
    
    // cout << dp[x][y][0] << endl;
    
    if (dp[x][y][0] != 1)
        return "impossible";
    
    while (sk < k)
    {
        // cout << x << ' ' << y << ' ' << sk << endl;
        bool flag = false;
        for (int i = 0; i < 4; i++)
        {
            int nx = x + dx[i], ny = y + dy[i];
            if (nx <= 0 || nx <= 0 || nx > n || ny > m) continue;
            if (dp[nx][ny][sk + 1] == 1)
            {
                ans += ch[i];
                x = nx, y = ny, sk++;
                flag = true;
                break;
            }
        }
        if (!flag)
            return "impossible";
    }
    
    return ans; 
}
profile
안녕하세요 aj4941 입니다.

0개의 댓글