-
[프로그래머스 / JS] Lv.1 키패드 누르기Algorithm/프로그래머스 2022. 12. 27. 16:22반응형
문제 설명
스마트폰 전화 키패드의 각 칸에 다음과 같이 숫자들이 적혀 있습니다.
이 전화 키패드에서 왼손과 오른손의 엄지손가락만을 이용해서 숫자만을 입력하려고 합니다.
맨 처음 왼손 엄지손가락은 * 키패드에 오른손 엄지손가락은 # 키패드 위치에서 시작하며, 엄지손가락을 사용하는 규칙은 다음과 같습니다.- 엄지손가락은 상하좌우 4가지 방향으로만 이동할 수 있으며 키패드 이동 한 칸은 거리로 1에 해당합니다.
- 왼쪽 열의 3개의 숫자 1, 4, 7을 입력할 때는 왼손 엄지손가락을 사용합니다.
- 오른쪽 열의 3개의 숫자 3, 6, 9를 입력할 때는 오른손 엄지손가락을 사용합니다.
- 가운데 열의 4개의 숫자 2, 5, 8, 0을 입력할 때는 두 엄지손가락의 현재 키패드의 위치에서 더 가까운 엄지손가락을 사용합니다.
4-1. 만약 두 엄지손가락의 거리가 같다면, 오른손잡이는 오른손 엄지손가락, 왼손잡이는 왼손 엄지손가락을 사용합니다.
순서대로 누를 번호가 담긴 배열 numbers, 왼손잡이인지 오른손잡이인 지를 나타내는 문자열 hand가 매개변수로 주어질 때, 각 번호를 누른 엄지손가락이 왼손인 지 오른손인 지를 나타내는 연속된 문자열 형태로 return 하도록 solution 함수를 완성해주세요.
제한사항
- numbers 배열의 크기는 1 이상 1,000 이하입니다.
- numbers 배열 원소의 값은 0 이상 9 이하인 정수입니다.
- hand는 "left" 또는 "right" 입니다.
- "left"는 왼손잡이, "right"는 오른손잡이를 의미합니다.
- 왼손 엄지손가락을 사용한 경우는 L, 오른손 엄지손가락을 사용한 경우는 R을 순서대로 이어붙여 문자열 형태로 return 해주세요.
풀이
function solution(numbers, hand) { let keyPad = [[1, 4, 7], [2, 5, 8, 0], [3, 6, 9]]; let result = ''; let yOfCur = []; numbers.reduce(([posL, posR], cur) => { yOfCur = keyPad.map(ele => ele.indexOf(cur) + 1); if (yOfCur[0]) { posL = [-1, yOfCur[0]]; result += 'L'; } if (yOfCur[2]) { posR = [-1, yOfCur[2]]; result += 'R'; } if (yOfCur[1]) { let posCur = [0, yOfCur[1]]; let len = {'LtoNew': Math.abs(posL[0]) + Math.abs(posL[1] - posCur[1]), 'RtoNew': Math.abs(posR[0]) + Math.abs(posR[1] - posCur[1])}; if (len.LtoNew > len.RtoNew || ((len.LtoNew === len.RtoNew) && hand === 'right')) { posR = [...posCur]; result += 'R'; } if (len.LtoNew < len.RtoNew || ((len.LtoNew === len.RtoNew) && hand === 'left')) { posL = [...posCur]; result += 'L'; } } return [posL, posR]; }, [[-1, 4], [1, 4]]) return result; }
실행에 걸린 최대 시간 : 1.17ms
사고과정
- 키패트를 좌표계로 생각하고 풀어야겠다.
- 새로 입력되는 숫자의 좌표와 현재 손가락의 위치 좌표가 필요하겠어
- 새로 입력된 숫자가 [1, 4, 7], [2, 5, 8, 0], [3, 6, 9] 중 첫번째, 세번재 경우에 포함되면 바로 결과값에 L 또는 R을 추가하고 해당 방향 손가락의 좌표를 새로 입력된 숫자의 좌표로 갱신하자
- 위에서 두 번째 경우 [2, 5, 8, 0]에 포함되는 경우 새로 입력된 숫자부터 현재 각 손가락까지의 거리가 필요해
- 상, 하, 좌, 우로만 움직임이 가능하니 새로입력된 숫자의 좌표와 현재 각 손가락들의 좌표를 각 인덱스기릴 배준 후 절댓값을 씌워주면 새로 입력된 숫자에서 부터 양 손가락 까지의 거리를 구할 수 있다.
- 윈손이 가까운 경우와 거리가 같은데 왼손잡이인 경우, 오른손이 가까운 경우와 거리가 같은데 오른손 잡이인 경우로 나누어서 결가값 추가 및 좌표값 갱신을 진행하자
- 음, 성공은 하는데 코드가 지저분해
- 3에서 조건에서 include를 쓰고 조건문 내부에서 indexOf를 쓰네.. 비효율적이야
- 이터레이션 외부에 yOfCur이란 변수를 생성하고 여기에 입력된 숫자가 [1, 4, 7], [2, 5, 8, 0], [3, 6, 9] 각각에서 어떤 인덱스를 갖는지 저장하자.
- 어차피 조건으로 쓸거니까 [1, 4, 7], [2, 5, 8, 0], [3, 6, 9]에 포함되지 않을 경우 0을 가지면 falsy라 보기 편하겠어 keyPad.map의 콜백에서 indexOf를 쓴 다음에 1을 더하자
- 빨라지긴 했는 각 조건문에서 실행되는 코드가 아직 중복된것 같단 말이지 이건 축약 못 시키나???
- 추가적으로 키 값을 생성해서 축약시킬 수 있겠는데 코드는 조금 짧아지겠지만 이해하기에도 불편할 것 같고, 굳이 변수를 추가할 필요도 없을 것 같으니 굳이 축약시킬 필요는 없겠다.
평가
개선점
알게된점
반응형'Algorithm > 프로그래머스' 카테고리의 다른 글
[프로그래머스 / JS] Lv.1 실패율 (0) 2022.12.27 [프로그래머스 / JS] Lv.1 크레인 인형뽑기 게임 (0) 2022.12.27 [프로그래머스 / JS] Lv.1 3진법 뒤집기 (0) 2022.12.26 [프로그래머스 / JS] Lv.1 신규 아이디 추천 (0) 2022.12.26 [프로그래머스 / JS] Lv.1 로또 최고 순위와 최저 순위 (0) 2022.12.26