일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
Tags
- 코드트리빵
- 슈퍼컴퓨터클러스터
- 토끼와 경주
- 백준
- DP
- ICER
- 루돌프의반란
- ros
- 소프티어
- 구현
- 포탑부수기
- BFS
- 순서대로방문하기
- 이진탐색
- DenseDepth
- dfs
- 마법의숲탐색
- 싸움땅
- 3Dreconstruction
- ISER
- 나무박멸
- 조합
- Calibration
- 삼성기출
- 시뮬레이션
- 코드트리
- ARM
- 마이크로프로세서
- 수영대회결승전
- 왕실의기사대결
Archives
- Today
- Total
from palette import colorful_colors
[프로그래머스] 주사위 고르기 with C++ (2024 카카오 겨울 인턴십) 본문
접근방법
1. 조합짜기
먼저 주사위 중 A가 굴릴 주사위를 정할 조합을 짠다. 그러면 B가 굴릴 주사위도 자동으로 정해진다.
A가 선택한 주사위는 aSelected가 1이고, B가 선택한 주사위는 aSelected가 0이다. → 재귀 이용, makePath 함수
2. 해당 조합마다 굴리는 경우 구하기
A가 굴리는 주사위의 합, B가 굴리는 주사위 합만 알면 되므로, (6^N) 이 아닌 (6^N/2) 를 두 번 구해주면 된다.
굴리는 주사위의 합들을 각각의 DAT에 저장하고, 나중에 비교를 위해 합이 나오는 경우의 수를 aCase, bCase에 저장한다. → 재귀 이용, roll함수
3. 해당 조합에서 A가 이기는 경우의 수 구하기
비교시 시간복잡도를 줄이기 위해 aCase, bCase를 오름차순 정렬, 중복제거했다. (vector의 unique 활용)
이후 aCase의 원소마다, 그것보다 작은 bCase원소의 개수를 구하고, 해당 aCase 원소가 나오는 경우의 수 x bCase원소가 나오는 경우의 수를 구하면 해당 조합에서 A가 이기는 경우의 수가 구해진다! (DAT 활용)
이진탐색이나 누적합으로 구할 수도 있다. 그리고 그게 더 빨라 보인다🥲
최종코드
#include <iostream>
#include <string>
#include <vector>
#include <cstring>
#include <algorithm>
using namespace std;
vector<int> answer;
vector<vector<int>> diceMAP; // 주사위들 전역변수로 저장
int diceCnt, maxWinCnt; // 주사위 개수 (2~ 10), A가 해당 조합에서 최대 이긴 횟수
vector <int> aPath, bPath; // A, B에 해당하는 조합 저장
vector <int> aCase, bCase; // 각 조합 당 합의 경우를 저장
int aSelected[10];
int aDAT[501];
int bDAT[501];
// A 또는 B 해당 path(조합) 마다 굴러서 주사위 합 저장하기
void roll(int level, int sum, int isA, vector <int> path ){
if (level == diceCnt/2){
if (isA == 0){ // A 굴리는 경우
aDAT[sum] ++;
aCase.push_back(sum);
}
else{ // B 굴리는 경우
bDAT[sum]++;
bCase.push_back(sum);
}
return;
}
for (int i=0; i<6; i++){
roll(level + 1, sum + diceMAP[path[level]][i], isA, path);
}
}
void check(){
// 초기화
memset(aDAT, 0, sizeof(aDAT));
memset(bDAT, 0, sizeof(bDAT));
aCase.clear();
bCase.clear();
// A가 가져감에 따라 자동으로 선택되는 B 주사위 path 만들기
bPath.clear();
for (int i=0; i<diceCnt; i++){
if (aSelected[i] == 0) bPath.push_back(i);
}
// DAT, Case에 값 저장
roll(0, 0, 0, aPath);
roll(0, 0, 1, bPath);
// aCase, bCase에 정렬, 중복 제거
sort(aCase.begin(),aCase.end());
aCase.erase(unique(aCase.begin(),aCase.end()),aCase.end());
sort(bCase.begin(),bCase.end());
bCase.erase(unique(bCase.begin(),bCase.end()),bCase.end());
// 경우의 수 계산
int curWinCnt = 0, bCnt = 0;
for (int i = 0; i < aCase.size(); i++) {
int curCnt = 0;
// 현재 aCase 원소보다 작은 bCase 원소 개수를 센다.
while (bCnt < bCase.size() && bCase[bCnt] < aCase[i]) {
bCnt++;
}
// 현재 aCase 원소 DAT x bCase 원소 DAT
for (int j=0; j< bCnt; j++){
curWinCnt += (aDAT[aCase[i]] * bDAT[bCase[j]]);
}
}
// 정답 업데이트
if (curWinCnt > maxWinCnt){
maxWinCnt = curWinCnt;
answer.clear();
for(int i=0; i<aPath.size(); i++){
answer.push_back(aPath[i] + 1);
}
}
}
// 조합 만들기 함수 -> path에 저장
void makePath(int level, int pre){
if (level == diceCnt/2){
check();
return;
}
for (int i=pre+1; i < diceCnt; i++){
aPath.push_back(i);
aSelected[i] = 1;
makePath(level+1, i);
aSelected[i] = 0;
aPath.pop_back();
}
}
vector<int> solution(vector<vector<int>> dice) {
diceMAP = dice;
diceCnt = dice.size();
makePath(0, -1);
return answer;
}
'알고리즘 > 문제풀이' 카테고리의 다른 글
[Softeer] 슈퍼컴퓨터 클러스터 with C++ (HSAT 4회 기출) (0) | 2024.05.01 |
---|---|
[백준] RGB거리 with C++ (1) | 2024.05.01 |
[백준] 1715 카드 정렬하기 with C++ (1) | 2024.05.01 |
[코드트리] 예술성 with C++ (삼성기출) (0) | 2024.04.21 |
[코드트리] 마법의 숲 탐색 with C++ (삼성기출) (1) | 2024.04.19 |