오늘의 인기 글
최근 글
최근 댓글
Today
Total
01-08 01:48
관리 메뉴

우노

[DFS] 백준 1987번 알파벳 C++ 풀이 본문

Algorithm/DFS

[DFS] 백준 1987번 알파벳 C++ 풀이

운호(Noah) 2021. 7. 6. 14:09

문제 링크

풀이

  • 이미 방문한 위치를 제외하고, 상하좌우로 최대한 얼마만큼 갈 수 있는지에 대한 문제이다.
  • 방문 여부는 'alphabet' 1차원 배열을 사용해 저장했다.
    • 알파벳은 'A' 부터 'Z' 까지의 문자로 이루어져있으며,
    • 각 문자를 int형으로 변환하면, 10진수 65 ~ 90 의 값으로 이루어져있다.
    • 따라서, 'A' 의 경우 int형 변환 후, 65를 빼면 0 이기 때문에, 0번째 index 를 'A'의 위치로 설정하고
    • 'Z' 는 65 를 빼면 25 이기 때문에, 25번째 index 를 'Z'의 위치로 설정했다.
  • 그리고 dfs 함수를 호출하기 전에, 방문한 지점의 알파벳을 방문한 것으로 체크하고
  • dfs 함수를 호출한 뒤에는, 방문한 지점의 알파벳을 방문하지 않은 것으로 체크했다.
    • 다른 경로로 가는 경우도 모두 구하기 위해서이다.

코드

#include <iostream>
#include <algorithm>
#define MAX 20
using namespace std;

// 입력
int r, c;

// 알파벳을 담고 있는 보드 행렬
char board[MAX][MAX];

// 방문한 알파벳
int alphabet[26] = {0};

// row index 변화 목록 (상, 하, 좌, 우)
int dx[4] = {-1, 1, 0, 0};

// col index 변화 목록 (상, 하, 좌, 우)
int dy[4] = {0, 0, -1, +1};

// 말이 지날 수 있는 최대 경로 길이
int max_path = 0;

// 상하좌우 경로 탐색 (행, 열, 현재까지 찾은 경로의 길이)
void dfs(int row, int col, int find_path){

    // 현재까지 찾은 경로가 max_path보다 길다면, max_path 수정
    max_path = max(find_path, max_path);

    // 상하좌우에 따라 row, col 값 변경
    for (int i=0; i<4; ++i){
        int new_row = row + dx[i];
        int new_col = col + dy[i];

        // 상하좌우로 움직일 수 있는지 확인
        if (0 <= new_row && new_row < r && 0 <= new_col && new_col < c){

            // 이동할 지점의 알파벳이 이전에 방문한 알파벳인지 확인한 후, 방문 가능하다면
            if (!alphabet[((int)board[new_row][new_col])-65]){

                // 현재 위치의 알파벳을 방문한 알파벳으로 저장
                alphabet[((int)board[new_row][new_col])-65]++;

                // 해당 정보들을 가지고 dfs 탐색 시작
                dfs(new_row, new_col, find_path+1);    

                // 상단의 재귀함수가 끝났다면, 인접한 다른 경로도 확인해야하므로
                // 현재 위치의 알파벳을 방문하지 않은 알파벳으로 변경
                alphabet[((int)board[new_row][new_col])-65]--;

            }
        }
    }
}

int main(void){

    // r, c 입력
    cin >> r >> c;

    // 알파벳을 입력 받아 보드에 저장
    for(int i=0; i<r; ++i){
        for (int j=0; j<c; ++j){
            cin >> board[i][j];
        }
    }

    // 0,0 은 방문한 알파벳으로 저장
    alphabet[((int)board[0][0])-65]++;

    // 경로 탐색 함수 호출
    dfs(0,0,1);

    // 출력
    cout << max_path << endl;

    return 0;

}

참고

Comments