문제설명 :
카카오톡 게임별의 하반기 신규 서비스로 다트 게임을 출시하기로 했다. 다트 게임은 다트판에 다트를 세 차례 던져 그 점수의 합계로 실력을 겨루는 게임으로, 모두가 간단히 즐길 수 있다.
갓 입사한 무지는 코딩 실력을 인정받아 게임의 핵심 부분인 점수 계산 로직을 맡게 되었다. 다트 게임의 점수 계산 로직은 아래와 같다.
0~10의 정수와 문자 S, D, T, *, #로 구성된 문자열이 입력될 시 총점수를 반환하는 함수를 작성하라.
제한 사항
"점수|보너스|[옵션]"으로 이루어진 문자열 3세트.
예) 1S2D*3T
입출력 예
dartResult | Return |
"1S2D*3T" | 37 |
"1D2S#10S" | 9 |
"1D2S0T" | 3 |
"1S*2T*3S" | 23 |
"1D#2S*3S" | 5 |
"1T2D3D#" | -4 |
"1D2S3T*" | 59 |
첫번째 접근
: 숫자와 문자가 붙어있는 형태이므로 먼저 이를 분리해 구분하고자 했다. 그래서 dartResult string을 split 하여 배열을 만든 후 각 요소를 돌며 해당 요소가 문자일땐, str에 공백과 함께 해당 문자를 더해주었고 해당 요소가 숫자로 변환되거나 '0'일 때, str의 가장 마지막에 있는 문자가 숫자로 변경되는지 확인한 후, 숫자이면 str에 바로 숫자로 변환가능한 문자를 붙여주었고, 그렇지 않고 str의 마지막 문자가 숫자로 변환되지 않는 경우 공백과 함께 숫자를 넣어주었다. 이는 '10'과 같이 숫자가 한 자리 이상일 때의 예외처리를 위함이다.
그럼 str은 공백으로 시작하게 되므로 0번 인덱스의 공백을 지워준 후 split하여 배열을 만들어주었다. 이후 이 배열을 돌며 해당 요소가 숫자이면 nums 배열에 해당 요소를 숫자로 변경해 넣어주었다. 해당 요소가 0인 경우는 Number(0)은 false이므로 따로 처리해주었다. 이후 과정은 문제의 조건에 맞게 처리했다.
function solution(dartResult) { // "1D2S#10S"
let str = '' // ' 1 D 2 S # 10 S'
dartResult.split('').forEach(el => {
if(Number(el) || el === '0') {
if(Number(str.slice(str.length - 1))) str += el
else str += ' ' + el
}
else str += ' ' + el
})
let nums = []
str.slice(1).split(' ').forEach(el => {
if(Number(el)) nums.push(Number(el))
if(el === '0') nums.push(0)
else {
let target = nums[nums.length - 1]
if(el === 'D') nums[nums.length - 1] = Math.pow(target, 2)
if(el === 'T') nums[nums.length - 1] = Math.pow(target, 3)
if(el === '*'){
if(nums.length === 1) nums[0] = target * 2
else {
nums[nums.length - 2] = nums[nums.length - 2] * 2
nums[nums.length - 1] = target * 2
}
}
if(el === '#') nums[nums.length - 1] = target * -1
}
})
return nums.reduce((a, c) => a + c)
}
두번째 접근
: 코드가 다소 장황한것 같아 switch 문을 사용한 코드로 변경해주었다. 또한 str의 값을 생성하는 과정에서도 코드가 중복되는 것 같아 다듬어주었다.
function solution(dartResult) {
let str = ''
dartResult.split('').map(el => {
if((Number(el) || el === '0') && Number(str.slice(str.length - 1))) str += el
else str += ' ' + el
})
let nums = []
str.slice(1).split(' ').forEach(el => {
let target = nums[nums.length - 1]
if(Number(el)) nums.push(Number(el))
switch (el){
case '0' :
nums.push(0);
break;
case 'D' :
nums[nums.length - 1] = Math.pow(target, 2);
break;
case 'T' :
nums[nums.length - 1] = Math.pow(target, 3);
break;
case '*' :
if(nums.length === 1) nums[0] = target * 2
else {
nums[nums.length - 2] = nums[nums.length - 2] * 2
nums[nums.length - 1] = target * 2
};
break;
case '#':
nums[nums.length - 1] = target * -1;
break;
}
})
return nums.reduce((a, c) => a + c)
}
solution
unction solution(dartResult) {
const bonus = { 'S': 1, 'D': 2, 'T': 3 },
options = { '*': 2, '#': -1, undefined: 1 };
let darts = dartResult.match(/\d.?\D/g);
for (let i = 0; i < darts.length; i++) {
let split = darts[i].match(/(^\d{1,})(S|D|T)(\*|#)?/),
score = Math.pow(split[1], bonus[split[2]]) * options[split[3]];
if (split[3] === '*' && darts[i - 1]) darts[i - 1] *= options['*'];
darts[i] = score;
}
return darts.reduce((a, b) => a + b);
}
JS Practice) 프로그래머스 Lv.1 : 옹알이 (2) (0) | 2023.08.29 |
---|---|
JS Practice) 프로그래머스 Lv.1 : 로또의 최고 순위와 최저 순위 (0) | 2023.08.28 |
JS Practice) 프로그래머스 Lv.1 : 기사단원의 무기 (0) | 2023.08.24 |
JS Practice) 프로그래머스 Lv.1 : 실패율 (0) | 2023.08.23 |
JS Practice) 프로그래머스 Lv.1 : 덧칠하기 (0) | 2023.08.22 |
댓글 영역