반응형

🤔 문제


 

 

코딩테스트 연습 - 조이스틱

조이스틱으로 알파벳 이름을 완성하세요. 맨 처음엔 A로만 이루어져 있습니다. ex) 완성해야 하는 이름이 세 글자면 AAA, 네 글자면 AAAA 조이스틱을 각 방향으로 움직이면 아래와 같습니다. ▲ - 다

programmers.co.kr

 

😀 풀이


해당 문제는 프로그래머스의 그리디 유형으로 올라온 문제인데.. 테스트 케이스 추가 전에는 정답 처리가 되었다가 다시 채점해보니 틀렸다고 나와 재풀이한 문제입니다.

 

그리디로 풀었던 방법은 현재 위치를 기준으로 'A'가 아닌 곳 중 가장 짧은 곳으로 가서 변경하고 이를 반복하는 작업을 했습니다. 그런데 이런식으로 풀면 반례가 생깁니다...!

 

대표적으로 "BBBBAAAABA" 가 있습니다.

  • 그리디 방식을 사용할 경우 아래와 같이 움직여야 합니다.
    • >>><<<<< (8번)
  • 하지만, 실제 최소 좌우 움직임은 아래와 같습니다.
    • <<>>>>> (7번)

따라서 풀이 방식을 고민하다가 입력으로 주어지는 문자열의 최대 길이가 20이기 때문에 완전 탐색을 고려해보게 되었고, 이를 통해 정답을 얻었습니다.

 

해당 문제를 풀때 편의를 위해 두 개의 추가 함수를 만들었습니다.

  • 알파벳이 주어졌을 때 상,하로 움직여야 하는 횟수를 구하는 함수 (countChange
  • 현재 위치에서 목표 위치로 갈 때 최소로 가려면 왼쪽으로 가야 하는지, 오른쪽으로 가야 하는지 확인하고 둘 중 최단 경로의 거리가 얼마인지 구하는 함수 (findShortestPath

# 완전 탐색

완전 탐색 방식은 다음과 같습니다.

  • 가야할 곳은 'A'가 아닌 곳입니다. 
  • 따라서 'A'가 아닌 곳의 인덱스를 전부 찾은 뒤(시작 위치 제외) 해당 인덱스들을 모두 방문하는 모든 경우의 수, 즉 순열을 구합니다.
  • 각 순열에 따른 이동 거리를 구해보고 이를 가지고 정답을 갱신합니다.
  • 모든 순열 중 최소 이동 거리가 나온 것이 문제에서 요구한 정답이 됩니다.

💻 풀이 코드는 아래와 같습니다.

from itertools import permutations as p

INF = int(1e9)

# 알파벳이 주어졌을 때 상하로 움직이는 횟수 구하는 함수
def countChange(alp):
    return min(ord('Z') - ord(alp) + 1, ord(alp) - ord('A'))

# 왼쪽, 오른쪽 중 최단으로 가는 거리 구하는 함수
def findShortestPath(name, now, next):
    right, left = max(next, now), min(next, now)
    rightDist = right - left
    leftDist = left + len(name) - right
    return min(rightDist, leftDist)

def solution(name):
    answer = INF
    # "A" 가 아니라서 가야하는 위치(시작 위치 제외)
    toGoPlaces = [i for i in range(len(name)) if name[i] != "A" and i != 0]

    # 알파벳을 바꾸느라 생기는 이동
    changeCount = 0
    for c in name:
        changeCount += countChange(c);

    # 움직일 수 있는 모든 케이스
    cases = p(toGoPlaces, len(toGoPlaces))
    for case in cases:
        now = 0
        result = 0

        for next in case:
            dist = findShortestPath(name, now, next)
            result += dist
            now = next
            
        answer = min(answer, result + changeCount)
    return answer
복사했습니다!