Programers : 방의 개수 - C++

김정욱·2021년 3월 19일
0

Algorithm - 문제

목록 보기
168/249
post-thumbnail
post-custom-banner

방의 개수

코드

[ DFS 풀이 ]

#include <string>
#include <vector>
#include <map>
#include <iostream>
#include <algorithm>
using namespace std;
// 2차원 좌표로 이루어진 그래프에서 순환 검사 + 예외처리
map<pair<int,int>, vector<pair<int,int>>> m;
map<pair<int,int>, bool> vis;
map<pair<pair<int,int>, pair<int,int>>, bool> make;
int dx[8] = {0, 1, 1, 1, 0, -1, -1, -1};
int dy[8] = {-1, -1, 0, 1, 1, 1, 0, -1};
int ans=0;
void DFS(int Y, int X){
    vis[{Y,X}] = true;
    for(int i=0;i<m[{Y,X}].size();i++){
        int ny = m[{Y,X}][i].first;
        int nx = m[{Y,X}][i].second;
        bool check = false;
        if(make[{{Y,X}, {ny,nx}}] or make[{{ny,nx}, {Y,X}}]) check = true;
        /* 해당 정점에 대해 방문하였으면서 && 처음 만들어지는 간선이면! */
        if(vis[{ny,nx}] and !check) ans++;
        else if(vis[{ny,nx}]) continue;
        /* 간선은 하나지만, 방향은 2개가 될 수 있기에 2방향 모두 넣어줘야함 */
        make[{{Y,X}, {ny,nx}}] = true;
        make[{{ny,nx}, {Y,X}}] = true;
        DFS(ny, nx);
    }
}
int solution(vector<int> arrows) {
    int curX = 0;
    int curY = 0;
    for(int i=0;i<arrows.size();i++)
    {
        /* 교차 예외 처리를 위해 2번식 이동 */
        for(int j=0;j<2;j++)
        {
            int nx = curX + dx[arrows[i]];
            int ny = curY + dy[arrows[i]];
            bool flag = false;
            m[{curY, curX}].push_back({ny, nx});
            curY = ny;
            curX = nx;
        }
    }
    DFS(0,0);
    return ans;
}
  • 핵심(방을 만드는 조건)
    • 이미 방문한 정점다시 방문하는 경우(vis)
    • 간선이 처음 만들어 지는 경우(make)
  • 예외 처리(교차)
    • 교차시 정점으로 체크할 수 없는 2개의 방이 만들어짐
    • 중간 지점정점으로 만들면 교차점을 검사할 수 있음
      --> 이동2번으로 나누어서 진행해야 함
  • 결과
    : 재귀로 해결해서 시간이좀 걸리지만 통과는 함 (반복문으로 하면 더 빠름)

[ 반복문 풀이 ]

profile
Developer & PhotoGrapher
post-custom-banner

0개의 댓글