프로그래밍/알고리즘
프로그래머스/해시/위장
Doublsb
2021. 8. 7. 00:10
https://programmers.co.kr/learn/courses/30/lessons/42578
풀이
- 스파이들은 매번 다른 옷으로 조합해서 입어야 함 (그냥 하나의 항목만 다르면 됨)
- 놀랍게도 최소 한 부위만 입으면 되므로, 티셔츠만 하나 입는 것도 가능함 :O.... 그게 스파이냐
- 서로 다른 옷 조합의 수를 return하도록 짜기
- Input은 이런 방식. ["crowmask", "face"]
- 'face' 키에 밸류로 리스트를 넣으면 되겠군
- 키 하나, 둘, 셋, ..., N개를 뽑아서 조합을 시도하면 되겠군
- 조합을 시도할 때 해당 항목에 리스트의 개수를 생각해서 넣으면 되겠군
C#
using System;
using System.Collections.Generic;
using System.Linq;
public class Solution {
public int solution(string[,] clothes) {
var answer = 0;
var closet = new Dictionary<string, List<string>>();
//옷장에 정리해두기
for(int i=0; i<clothes.GetLength(0); i++)
{
if(!closet.ContainsKey(clothes[i, 1]))
closet.Add(clothes[i, 1], new List<string>());
closet[clothes[i, 1]].Add(clothes[i, 0]);
}
//최소 1개 이상의 종류를 뽑아서 입기
for(int i=1; i<=closet.Keys.Count; i++)
{
var combination_of_types = new CombinationSet(closet.Keys.ToList(), i).result;
var result = 1;
foreach(var e in combination_of_types)
{
foreach(var type in e) {
result *= closet[type].Count;
}
answer += result;
result = 1;
}
}
return answer;
}
public class CombinationSet
{
public List<List<string>> result;
private List<string> elements;
public CombinationSet(List<string> elements, int count)
{
result = new List<List<string>>();
this.elements = elements;
get_combination(count, new List<string>(), 0);
}
private void get_combination(int count, List<string> tmp, int index)
{
if(count == 0)
{
result.Add(new List<string>(tmp));
return;
}
for(int i=index; i<elements.Count; i++)
{
tmp.Add(elements[i]);
get_combination(count - 1, tmp, i+1);
tmp.RemoveAt(tmp.Count - 1);
}
}
}
}
결과
정확성 96.4/100
피드백
1번 테스트케이스에서만 시간 초과.
시간 초과는 극단적인 케이스에서 발생한다. 예를 들어, 30종류의 옷을 1가지씩 가지고 있으면 무한으로 조합을 돌리게 된다 (...)
다시 풀어본다.
풀이
- 매우 쉬워지는 방법이 있다. 각 파츠에 Naked 상태를 추가해보자. 이러면 그냥 순열이 되는 마법
- 그래도 전부 Naked일 수는 없으니 결과값에서 1만 빼준다.
using System;
using System.Collections.Generic;
public class Solution {
public int solution(string[,] clothes) {
var answer = 1;
var closet = new Dictionary<string, List<string>>();
//옷장에 정리해두기
for(int i=0; i<clothes.GetLength(0); i++)
{
if(!closet.ContainsKey(clothes[i, 1]))
closet.Add(clothes[i, 1], new List<string>());
closet[clothes[i, 1]].Add(clothes[i, 0]);
}
//각 키에 Naked 상태 추가하기
foreach(var type in closet.Keys){
closet[type].Add("Naked");
}
//한 파츠씩 뽑아서 입기. 순열이 되어버렸다
foreach(var type in closet.Keys){
answer *= closet[type].Count;
}
//다 벗고 돌아다닐 순 없는걸
return answer - 1;
}
}
결과
정확성 100/100
피드백
사실 Dictionary도 <string, string>일 필요가 없다. 가짓수를 출력하는 것이지 데이터를 출력하는 것이 아니니까.
<string, int>로 해시를 만들면 충분하긴 한데, 굳이 그 정도로 바꿀 필요성은 느끼지 못함.
반응형