2022. 9. 8. 15:46ㆍ알고리즘/programmers
레벨 1 처음 다루는 거 같습니다.
근데 생각보다 어려워요(레벨 1치고는)
같이 보죠!
문제 요약
나만의 카카오 성격 유형 검사지를 만들려고 합니다.
성격 유형 검사는 다음과 같은 4개 지표로 성격 유형을 구분합니다. 성격은 각 지표에서 두 유형 중 하나로 결정됩니다.
지표 번호성격 유형
1번 지표 | 라이언형(R), 튜브형(T) |
2번 지표 | 콘형(C), 프로도형(F) |
3번 지표 | 제이지형(J), 무지형(M) |
4번 지표 | 어피치형(A), 네오형(N) |
4개의 지표가 있으므로 성격 유형은 총 16(=2 x 2 x 2 x 2)가지가 나올 수 있습니다. 예를 들어, "RFMN"이나 "TCMA"와 같은 성격 유형이 있습니다.
검사지에는 총 n개의 질문이 있고, 각 질문에는 아래와 같은 7개의 선택지가 있습니다.
- 매우 비동의
- 비동의
- 약간 비동의
- 모르겠음
- 약간 동의
- 동의
- 매우 동의
각 질문은 1가지 지표로 성격 유형 점수를 판단합니다.
예를 들어, 어떤 한 질문에서 4번 지표로 아래 표처럼 점수를 매길 수 있습니다.
선택지 | 성격 유형 점수 |
매우 비동의 | 네오형 3점 |
비동의 | 네오형 2점 |
약간 비동의 | 네오형 1점 |
모르겠음 | 어떤 성격 유형도 점수를 얻지 않습니다 |
약간 동의 | 어피치형 1점 |
동의 | 어피치형 2점 |
매우 동의 | 어피치형 3점 |
이때 검사자가 질문에서 약간 동의 선택지를 선택할 경우 어피치형(A) 성격 유형 1점을 받게 됩니다. 만약 검사자가 매우 비동의 선택지를 선택할 경우 네오형(N) 성격 유형 3점을 받게 됩니다.
위 예시처럼 네오형이 비동의, 어피치형이 동의인 경우만 주어지지 않고, 질문에 따라 네오형이 동의, 어피치형이 비동의인 경우도 주어질 수 있습니다.
하지만 각 선택지는 고정적인 크기의 점수를 가지고 있습니다.
- 매우 동의나 매우 비동의 선택지를 선택하면 3점을 얻습니다.
- 동의나 비동의 선택지를 선택하면 2점을 얻습니다.
- 약간 동의나 약간 비동의 선택지를 선택하면 1점을 얻습니다.
- 모르겠음 선택지를 선택하면 점수를 얻지 않습니다.
검사 결과는 모든 질문의 성격 유형 점수를 더하여 각 지표에서 더 높은 점수를 받은 성격 유형이 검사자의 성격 유형이라고 판단합니다. 단, 하나의 지표에서 각 성격 유형 점수가 같으면, 두 성격 유형 중 사전 순으로 빠른 성격 유형을 검사자의 성격 유형이라고 판단합니다.
질문마다 판단하는 지표를 담은 1차원 문자열 배열 survey와 검사자가 각 질문마다 선택한 선택지를 담은 1차원 정수 배열 choices가 매개변수로 주어집니다. 이때, 검사자의 성격 유형 검사 결과를 지표 번호 순서대로 return 하도록 solution 함수를 완성해주세요.
제한사항
- 1 ≤ survey의 길이 ( = n) ≤ 1,000
- survey의 원소는 "RT", "TR", "FC", "CF", "MJ", "JM", "AN", "NA" 중 하나입니다.
- survey[i]의 첫 번째 캐릭터는 i+1번 질문의 비동의 관련 선택지를 선택하면 받는 성격 유형을 의미합니다.
- survey[i]의 두 번째 캐릭터는 i+1번 질문의 동의 관련 선택지를 선택하면 받는 성격 유형을 의미합니다.
- choices의 길이 = survey의 길이
- choices[i]는 검사자가 선택한 i+1번째 질문의 선택지를 의미합니다.
- 1 ≤ choices의 원소 ≤ 7
테스트케이스
survey | choices | result |
["AN", "CF", "MJ", "RT", "NA"] | [5, 3, 2, 7, 5] | "TCMA" |
["TR", "RT", "TR"] | [7, 1, 3] | "RCJA" |
접근 방법
대충보기엔 쉬운데 역방향인게 문제입니다.
TR, RT를 같은 key로 정리해주는 노하우를 정리하는게 포인트에요.
다양한 방법이 있겠지만, 전 그냥 초기화할 때 나름의 법칙을 두고 만들어줬습니다. (별건 없습니다.. 하하)
survey의 원소는 "RT", "TR", "FC", "CF", "MJ", "JM", "AN", "NA" 중 하나입니다.
key에 무조건 오름차순을 넣어주고, map.has(key)를 했을 때 값이 없으면 역방향이라고 생각해주고 key를 뒤집어 주면 됩니다. 코드로 보시죠!
정답 코드
function solution(survey, choices) {
const mbti = new Map();
const combis = ["RT", "TR", "CF", "FC", "JM", "MJ", "AN", "NA"];
combis.forEach((v, i) => i % 2 === 0 && mbti.set(v, 0));
for (let i = 0; i < survey.length; i++) {
const type = survey[i];
const score = Math.abs(choices[i] - 4);
const isFirstType = choices[i] < 4;
if (mbti.has(type)) {
mbti.set(type, mbti.get(type) + (isFirstType ? score : -score));
} else {
const reverse = type.split("").sort().join("");
mbti.set(reverse, mbti.get(reverse) + (isFirstType ? -score : score));
}
}
return [...mbti].map(([key, val]) => (val < 0) ? key[1] : key[0]).join("");
}
팁 1. TR을 RT로 어떻게 바꿔서 넣는지
if (mbti.has(type)) {
// 오름차순 type. 그냥 넣어줍니다.
mbti.set(type, mbti.get(type) + (isFirstType ? score : -score));
} else {
// 내림차순 type. score에 -를 곱해서 넣어줍시다.
const reverse = type.split("").sort().join("");
mbti.set(reverse, mbti.get(reverse) + (isFirstType ? -score : score));
}
별건 없고 그냥 값을 양수, 음수로 나눠서 관리해주면 됩니다.
아니 근데, 이거 다음 문제가 레벨2 "두 큐의 합 같게 하기"인데,
이거 너무 어려워요.. 뒤에 dp 문제나 다익스트라보다 더 어려운거같은데.. javascript 억까가 있네요..
아무튼 다음 문제에서 봅시다!
'알고리즘 > programmers' 카테고리의 다른 글
[JS] 숫자 게임 - 비겼을 때 처리를 해주자! (0) | 2022.09.14 |
---|---|
[JS] 최고의 집합: 산술-기하 평균으로 접근해보자! (0) | 2022.09.12 |
[JS] 코딩 테스트 공부 : 공부 팁 X, 카카오 2022 인턴 문제 O (0) | 2022.09.06 |
[JS] 가장 긴 팰린드롬 - dp를 잘 써보자! (1) | 2022.09.05 |
[JS] 경주로 건설 - 코테 인생 최초로 9000ms대를 만나다! (0) | 2022.09.03 |