[백준 c++] 2931 가스관

jw·2023년 7월 20일
0

백준

목록 보기
139/141
post-thumbnail

문제

러시아 가스를 크로아티아로 운반하기 위해 자그레브와 모스코바는 파이프라인을 디자인하고 있다. 두 사람은 실제 디자인을 하기 전에 파이프 매니아 게임을 이용해서 설계를 해보려고 한다.

이 게임에서 유럽은 R행 C열로 나누어져 있다. 각 칸은 비어있거나, 아래 그림과 같은 일곱가지 기본 블록으로 이루어져 있다.

블록 '|' 블록 '-' 블록 '+' 블록 '1' 블록 '2' 블록 '3' 블록 '4'
가스는 모스크바에서 자그레브로 흐른다. 가스는 블록을 통해 양방향으로 흐를 수 있다. '+'는 특별한 블록으로, 아래 예시처럼 두 방향 (수직, 수평)으로 흘러야 한다.

파이프 라인의 설계를 마친 후 두 사람은 잠시 저녁을 먹으러 갔다. 그 사이 해커가 침임해 블록 하나를 지웠다. 지운 블록은 빈 칸이 되어있다.

해커가 어떤 칸을 지웠고, 그 칸에는 원래 어떤 블록이 있었는지 구하는 프로그램을 작성하시오.

입력
첫째 줄에 유럽의 크기 R과 C가 주어진다. (1 ≤ R, C ≤ 25)

다음 R개 줄에는 C개 글자가 주어지며, 다음과 같은 글자로 이루어져 있다.

빈칸을 나타내는 '.'
블록을 나타내는 '|'(아스키 124), '-','+','1','2','3','4'
모스크바의 위치를 나타내는 'M'과 자그레브를 나타내는 'Z'. 두 글자는 한 번만 주어진다.
항상 답이 존재하고, 가스의 흐름이 유일한 경우만 입력으로 주어진다, 또, 모스크바와 자그레브가 하나의 블록과 인접해 있는 입력만 주어진다. 또, 불필요한 블록이 존재하지 않는다. 즉, 없어진 블록을 추가하면, 모든 블록에 가스가 흐르게 된다.

출력
지워진 블록의 행과 열 위치를 출력하고, 어떤 블록이었는지를 출력한다.

풀이

  1. M또는 Z에서 출발
  2. BFS를 이용하여 파이프 방향에 따라 이동
  3. 빈칸.을 만나면 탐색 종료
  4. 빈칸 좌표 중심으로 4방향 탐색해서 맞는 파이프 찾기

문제에서 분명 모스크바와 자그레브가 하나의 블록과 인접해 있는 입력만 주어진다
라고 쓰여있지만, 반드시 M또는 Z주변에 가스관이 하나도 없는 경우를 고려해야한다. 이걸 고려하지 않으면 3%에서 계속 틀렸습니다 뜸

코드

#include <iostream>
#include <algorithm>
#include <vector>
#include <queue>
using namespace std;
int r, c;
char map[26][26];
int dir[4][2] = {{1, 0}, {0, 1}, {-1, 0}, {0, -1}};
int open[4];
vector<pair<int, int>> m;
queue<pair<pair<int, int>, int>> q;
int ry, rx;

bool isValid(int y, int x)
{
    return (y >= 1 && y <= r) && (x >= 1 && x <= c);
}

void find_pipe()
{
    for (int i = 0; i < 4; i++)
    {
        int ny = ry + dir[i][0];
        int nx = rx + dir[i][1];
        char c = map[ny][nx];
        if (i == 0)
        {
            if (c == '|' || c == '+' || c == '2' || c == '3')
                open[0] = 1;
        }
        if (i == 1)
        {
            if (c == '-' || c == '+' || c == '3' || c == '4')
                open[1] = 1;
        }
        if (i == 2)
        {
            if (c == '|' || c == '+' || c == '1' || c == '4')
                open[2] = 1;
        }
        if (i == 3)
        {
            if (c == '-' || c == '+' || c == '1' || c == '2')
                open[3] = 1;
        }
    }
    cout << ry << " " << rx << " ";

    if (open[0] && open[1] && open[2] && open[3])
    {
        cout << "+";
    }
    else if (open[0] && !open[1] && open[2] && !open[3])
    {
        cout << "|";
    }
    else if (!open[0] && open[1] && !open[2] && open[3])
    {
        cout << "-";
    }
    else if (open[0] && open[1] && !open[2] && !open[3])
    {
        cout << "1";
    }
    else if (!open[0] && open[1] && open[2] && !open[3])
    {
        cout << "2";
    }
    else if (!open[0] && !open[1] && open[2] && open[3])
    {
        cout << "3";
    }
    else if (open[0] && !open[1] && !open[2] && open[3])
    {
        cout << "4";
    }
}
int check_dir(int y, int x)
{
    int d = -1;
    for (int i = 0; i < 4; i++)
    {
        int ny = y + dir[i][0];
        int nx = x + dir[i][1];

        char c = map[ny][nx];
        if (c == '.')
            continue;
        if (!isValid(ny, nx))
            continue;

        if (i == 0)
        {
            if (c == '|' || c == '+' || c == '2' || c == '3')
                d = 0;
        }
        else if (i == 1)
        {
            if (c == '-' || c == '+' || c == '3' || c == '4')
                d = 1;
        }
        else if (i == 2)
        {
            if (c == '|' || c == '+' || c == '1' || c == '4')
                d = 2;
        }
        else if (i == 3)
        {
            if (c == '-' || c == '+' || c == '1' || c == '2')
                d = 3;
        }
    }
    return d;
}

void search()
{
    int y = m[0].first;
    int x = m[0].second;
    int d = check_dir(y, x);

    if (d == -1)
    {
        y = m[1].first;
        x = m[1].second;
        d = check_dir(y, x);
    }
    q.push({{y, x}, d});

    while (q.size())
    {
        int y = q.front().first.first;
        int x = q.front().first.second;
        int d = q.front().second;
        int ny = y + dir[d][0];
        int nx = x + dir[d][1];
        int nd;
        char c = map[ny][nx];

        if (c == '.')
        {
            ry = ny;
            rx = nx;
            return;
        }

        if (d == 0)
        {
            if (c == '|' || c == '+')
                nd = 0;
            else if (c == '2')
                nd = 1;
            else if (c == '3')
                nd = 3;
        }
        else if (d == 1)
        {
            if (c == '-' || c == '+')
                nd = 1;
            else if (c == '3')
                nd = 2;
            else if (c == '4')
                nd = 0;
        }
        else if (d == 2)
        {
            if (c == '|' || c == '+')
                nd = 2;
            else if (c == '1')
                nd = 1;
            else if (c == '4')
                nd = 3;
        }
        else if (d == 3)
        {
            if (c == '-' || c == '+')
                nd = 3;
            else if (c == '1')
                nd = 0;
            else if (c == '2')
                nd = 2;
        }
        q.push({{ny, nx}, nd});
        q.pop();
    }
}

int main()
{
    ios_base::sync_with_stdio(0);
    cin.tie(0);
    cout.tie(0);

    cin >> r >> c;

    char s;
    for (int i = 1; i <= r; i++)
    {
        for (int j = 1; j <= c; j++)
        {
            cin >> s;
            map[i][j] = s;
            if (map[i][j] == 'M' || map[i][j] == 'Z')
                m.push_back({i, j});
        }
    }

    search();
    find_pipe();
}
profile
다시태어나고싶어요

1개의 댓글

comment-user-thumbnail
2023년 7월 20일

아주 유익한 내용이네요!

답글 달기

관련 채용 정보