728x90
문제 출처 : 프로그래머스 코딩테스트 연습 고득점 kit 해시 4번
( https://programmers.co.kr/learn/courses/30/parts/12077 )
문제 설명
스트리밍 사이트에서 장르 별로 가장 만힝 재생된 노래들 두 개씩 모아 베스트 앨범을 출시하려 한다.
노래는 고유번호(index로 구분)로 구분한다.
- 속한 노래가 많이 재생된 장르를 먼저 수록합니다.
- 장르 내에서 많이 재생된 노래를 먼저 수록합니다.
- 장르 내에서 재생 횟수가 같은 노래 중에서는 고유 번호가 낮은 노래를 먼저 수록합니다.
노래의 장르를 나타내는 문자열 배열 genres와 노래별 재생 횟수를 나타내는 정수 배열 plays가 주어질 때, 베스트 앨범에 들어갈 노래의 고유 번호를 순서대로 return 하도록 solution 함수를 완성하세요.
제한 사항
- genres[i]는 고유번호가 i인 노래의 장르입니다.
- plays[i]는 고유번호가 i인 노래가 재생된 횟수입니다.
- genres와 plays의 길이는 같으며, 이는 1 이상 10,000 이하입니다.
- 장르 종류는 100개 미만입니다.
- 장르에 속한 곡이 하나라면, 하나의 곡만 선택합니다.
- 모든 장르는 재생된 횟수가 다릅니다.
문제 풀이 방법
- 결과 반환용 빈 배열을 미리 선언
- 각 장르의 총 합을 모아둔 배열(playCntGenre)을 만들고 key와 value를 갖는 배열로 바꿔줌(Object.entries() 사용)
- key와 value를 갖는 배열을 sort로 내림 차순으로 정렬해준다.
- 장르와 재생수, 인덱스 정보를 갖는 객체 생성 (allInfoObj)
- allInfoObj에서 keyValueArr의 키값에 해당하는 장르들을 배열에 따로 저장해서 재생횟수
별로 내림차순 정렬을 한 후, 상위 두개 원소의 고유번호만 뽑아서 answer 배열에 push 해준다.
참고 문헌
- Object.entries() : https://developer.mozilla.org/ko/docs/Web/JavaScript/Reference/Global_Objects/Object/entries
- Array.prototype.sort() : https://developer.mozilla.org/ko/docs/Web/JavaScript/Reference/Global_Objects/Array/sort
- Array.prototype.reduce() : https://developer.mozilla.org/ko/docs/Web/JavaScript/Reference/Global_Objects/Array/Reduce
- Array.prototype.forEach() : https://developer.mozilla.org/ko/docs/Web/JavaScript/Reference/Global_Objects/Array/forEach
코드
function solution(genres, plays) {
let answer = [];
let playCntByGenre = {};
for(let i=0; i<genres.length; i++){
playCntByGenre[genres[i]] = playCntByGenre[genres[i]] ? playCntByGenre[genres[i]] + plays[i] : plays[i];
}; // 각 장르의 총 합을 정리
// playCntByGenre를 키와 값으로 변환 (entries메서드 이용)
let keyValueArr = Object.entries(playCntByGenre);
keyValueArr.sort((a,b) => b[1]-a[1]);
// 정렬함수의 compareFunc 작성시, return 값이 양수면 앞뒤순서가 바뀐다!
// keyValueArr는 [ [ 'pop', 3100 ], [ 'classic', 1450 ] ] 형태
let allInfoObj = genres.map((g, i) =>({
genre: g,
index: i,
playCnt: plays[i],
}));
// [
// { genre: 'classic', index: 0, playCnt: 500 },
// { genre: 'pop', index: 1, playCnt: 600 },
// { genre: 'classic', index: 2, playCnt: 150 },
// { genre: 'classic', index: 3, playCnt: 800 },
// { genre: 'pop', index: 4, playCnt: 2500 }
// ]
keyValueArr.forEach(k => {
let current = [];
for (let j = 0; j < allInfoObj.length; j++) {
if (k[0] === allInfoObj[j].genre) {
current.push(allInfoObj[j]);
};
};
current.sort((a, b) => b.playCnt - a.playCnt);
current.forEach((c, i) => {
if (i < 2) {
answer.push(c.index);
};
});
});
return answer;
}
주의점
sort()와 forEach()는 두번째가 현제 다루는 값의 인덱스값이다.
이걸 몰랐다....
728x90
'코딩 테스트 > 프로그래머스 코딩 테스트 연습' 카테고리의 다른 글
정렬 / 2 / 가장 큰 수 (JS) (0) | 2022.06.08 |
---|---|
정렬 / 1 / K번째 수 (JS) (0) | 2022.06.08 |
해시 / 2 / 위장 (JS) (0) | 2022.06.07 |
해시 / 2 / 전화번호 목록 (Java) (0) | 2022.06.07 |
해시 / 1 / 완주하지 못한 선수 (JS) (0) | 2022.06.07 |