기타
모각코_2회차
히어로맛쿠키
2024. 1. 12. 15:53
시뮬레이션 문제 풀기
https://www.acmicpc.net/problem/14889
나는
백트래킹1: N_C_2/N 조합 찾음 (halfCombination)
백트래킹2: 각 조합의 A, B 그룹 각각에서 2개씩 짝지어진 조합을 얻음 (getPairs)
그리고 점수를 계산해서 min값을 찾는 방법으로 했다.
그런데 또 다른친구 코드랑 비교해보니까.. splitTeam 백트래킹 안에서 모든 과정을 원콤에 끝낸다.
(원래 이런건 그렇게 하는 거라고 함)
메모리 10배이상, 시간 6배 ㅠㅠ
내가 백트래킹 감성을 모르고 있던 듯 하다... 다음 번에는 바로 반영해보는 걸로..
import java.util.*;
public class Main {
public static int N;
public static int[][] powers;
public static Map<String, int[][]> combinations = new HashMap<>();
public static int min = Integer.MAX_VALUE;
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
N = Integer.parseInt(sc.nextLine());
powers = new int[N+1][N+1];
for (int i = 1; i <= N; i++) {
String[] line = sc.nextLine().split(" ");
for (int j = 1; j <= N; j++) {
powers[i][j] = Integer.parseInt(line[j-1]);
}
}
halfCombination(0, 0, new int[N / 2]);
Set<String> keys = combinations.keySet();
for (String key : keys) {
int[][] combination = combinations.get(key);
int[] set1 = combination[0];
int[] set2 = combination[1];
// 2개씩 조합한 결과 리스트 얻기
List<int[]> pairs1 = getPairs(set1);
List<int[]> pairs2 = getPairs(set2);
int power1 = 0;
for (int[] pair : pairs1) {
int a = pair[0];
int b = pair[1];
power1 += powers[a][b] + powers[b][a];
}
int power2 = 0;
for (int[] pair : pairs2) {
int a = pair[0];
int b = pair[1];
power2 += powers[a][b] + powers[b][a];
}
int diff = Math.abs(power2 - power1);
if (diff < min) {
min = diff;
}
}
System.out.println(min);
}
public static List<int[]> getPairs(int[] set) {
List<int[]> result = new ArrayList<>();
getPairs(set, 0, 0, new int[2], result);
return result;
}
public static void getPairs(int[] set, int cnt, int setIdx, int[] picked, List<int[]> result) {
if (cnt == 2) {
result.add(picked);
return;
}
for (int i = setIdx; i < set.length; i++) {
int[] pickedClone = picked.clone();
pickedClone[cnt] = set[i];
getPairs(set, cnt + 1, i + 1, pickedClone, result);
}
}
public static void halfCombination(int idx, int element, int[] picked) {
if (combinations.containsKey(Arrays.toString(picked))) {
return;
}
if (idx == N / 2) {
int[] side = new int[N/2];
int i = 0;
for(int num=1; num<=N; num++){
boolean contained = false;
for (int p : picked) {
if (p == num) {
contained = true; break;
}
}
if (!contained){
side[i++] = num;
}
}
combinations.put(Arrays.toString(side), new int[][]{picked, side});
return;
}
for (int i = element+1; i <= N; i++) {
int[] pickedCopy = picked.clone();
pickedCopy[idx] = i;
halfCombination(idx + 1, i, pickedCopy);
}
}
}
반응형