조합 Combination (Java)
조합
조합이란 n
개의 숫자 중에서 r
개의 수를 순서 없이 뽑는 경우를 말합니다.
예를 들어 [1, 2, 3]
이란 숫자 배열에서 2개의 수를 순서 없이 뽑으면
[1, 2] [1, 3] [2, 3]
이렇게 3 개가 나옵니다.
순열을 뽑았을 때 나오는 [2, 1] [3, 1] [3, 2]
등은 중복이라서 제거됩니다.
부분 집합처럼 이해하시면 됩니다.
여러 가지 방법이 있지만 핵심은 하나입니다.
배열을 처음부터 마지막까지 돌며
- 현재 인덱스를 선택하는 경우
- 현재 인덱스를 선택하지 않는 경우
이렇게 두가지로 모든 경우를 완전탐색 해주시면 됩니다.
변수 | 설명 |
---|---|
arr | 조합을 뽑아낼 배열 |
output | 조합에 뽑혔는지 체크하는 배열 |
n | 배열의 길이 |
r | 조합의 길이 |
순열과 달리 조합은 r
을 유지할 필요가 없으므로 숫자를 하나 뽑을때마다 r
을 하나씩 줄여줍니다.
r == 0
일 때가 r
개의 숫자를 뽑은 경우입니다.
백트래킹 이용하여 구현
start
변수는 기준입니다.
start
인덱스를 기준으로 start
보다 작으면 뽑을 후보에서 제외, start
보다 크면 뽑을 후보가 됩니다.
예를 들어 [1, 2, 3]
배열이 있고 start
가 0 부터 시작합니다.
함수에 들어오면 먼저 i
가 start
부터 시작하는 for
문에 들어갑니다.
만약 0 인덱스인 1을 뽑는다면 visited[i]
는 true
가 되고 뽑지 않는다면 visited[i]
는 false
입니다.
1을 선택한 경우와 선택하지 않은 경우 둘 다 봐야합니다.
하지만 더이상 1은 고려 대상이 아니기 때문에 다음 for
문은 무조건 i+1
부터 시작해주어야 합니다.
Java Code
// 백트래킹 사용 // 사용 예시 : combination(arr, visited, 0, n, r) static void combination(int[] arr, boolean[] visited, int start, int n, int r) { if(r == 0) { print(arr, visited, n); return; } for(int i=start; i<n; i++) { visited[i] = true; combination(arr, visited, i + 1, n, r - 1); visited[i] = false; } }
Result
[4 개 중에서 1 개 뽑기] 1 2 3 4 [4 개 중에서 2 개 뽑기] 1 2 1 3 1 4 2 3 2 4 3 4 [4 개 중에서 3 개 뽑기] 1 2 3 1 2 4 1 3 4 2 3 4 [4 개 중에서 4 개 뽑기] 1 2 3 4
재귀 이용하여 구현
depth
변수를 사용합니다.
depth
변수는 현재 인덱스라고 생각하면 됩니다.
현재 인덱스를 뽑는다면 visited[depth] = true
뽑지 않는다면 visited[depth] = false
이렇게 진행하면 됩니다.
역시 마찬가지로 뽑은 경우와 뽑지 않은 경우 모두 봐야하고, 그 때 이전에 본 값들은 더이상 고려 대상이 아니기 때문에 depth
은 무조건 1 씩 증가합니다.
depth == n
이 되면 모든 인덱스를 다 보았으므로 재귀를 종료합니다.
Java Code
// 재귀 사용 // 사용 예시 : comb(arr, visited, 0, n, r) static void comb(int[] arr, boolean[] visited, int depth, int n, int r) { if (r == 0) { print(arr, visited, n); return; } if (depth == n) { return; } visited[depth] = true; comb(arr, visited, depth + 1, n, r - 1); visited[depth] = false; comb(arr, visited, depth + 1, n, r); }
Result
[4 개 중에서 1 개 뽑기] 1 2 3 4 [4 개 중에서 2 개 뽑기] 1 2 1 3 1 4 2 3 2 4 3 4 [4 개 중에서 3 개 뽑기] 1 2 3 1 2 4 1 3 4 2 3 4 [4 개 중에서 4 개 뽑기] 1 2 3 4
저는 개인적으로 재귀를 별로 좋아하지 않아서 백트래킹 방법을 주로 사용하는 편입니다.
어느 쪽이든 성능차이가 거의 없으니 편하고 익숙한 방법을 사용하시면 됩니다.
전체 소스코드
/** * 조합 : n 개 중에서 r 개 선택 */ public class Combination { public static void main(String[] args) { int n = 4; int[] arr = {1, 2, 3, 4}; boolean[] visited = new boolean[n]; for (int i = 1; i <= n; i++) { System.out.println("\n" + n + " 개 중에서 " + i + " 개 뽑기"); comb(arr, visited, 0, n, i); } for (int i = 1; i <= n; i++) { System.out.println("\n" + n + " 개 중에서 " + i + " 개 뽑기"); combination(arr, visited, 0, n, i); } } // 백트래킹 사용 // 사용 예시 : combination(arr, visited, 0, n, r) static void combination(int[] arr, boolean[] visited, int start, int n, int r) { if (r == 0) { print(arr, visited, n); return; } for (int i = start; i < n; i++) { visited[i] = true; combination(arr, visited, i + 1, n, r - 1); visited[i] = false; } } // 재귀 사용 // 사용 예시 : comb(arr, visited, 0, n, r) static void comb(int[] arr, boolean[] visited, int depth, int n, int r) { if (r == 0) { print(arr, visited, n); return; } if (depth == n) { return; } visited[depth] = true; comb(arr, visited, depth + 1, n, r - 1); visited[depth] = false; comb(arr, visited, depth + 1, n, r); } // 배열 출력 static void print(int[] arr, boolean[] visited, int n) { for (int i = 0; i < n; i++) { if (visited[i]) { System.out.print(arr[i] + " "); } } System.out.println(); } }
2차원 배열에서 조합
2차원 배열에서 조합을 뽑아야 할 때가 있어서 만들어보았습니다.
Java Code
/** * 2차원 배열에서 조합을 구할때 사용하는 방법 * 완전 탐색 할때 사용 */ public class Combination2 { static int n = 3; static int m = 3; public static void main(String[] args) { int[][] map = new int[n][m]; print(map); comb(map, 0, 0); } static void comb(int[][] map, int start, int depth) { if (depth == 3) { print(map); return; } for (int i = start; i < n * m; i++) { int x = i / m; int y = i % m; if (map[x][y] == 0) { map[x][y] = 1; comb(map, i + 1, depth + 1); map[x][y] = 0; } } } static void print(int[][] map) { for (int i = 0; i < n; i++) { for (int j = 0; j < m; j++) { System.out.print(map[i][j] + " "); } System.out.println(); } System.out.println(); } }