가능한 모든 경우를 탐색하는 중 해답으로 이어지지 않는 경우에 대해서 탐색하지 않고 되돌아가며 해결
<예시> N-Queen문제
<예시2> 민기의 햄버거 다이어트
import java.util.Scanner;
public class test {
static int N, L; // N : 재료의 개수, L : 제한 칼로리
static int[] cals;
static int[] scores;
static int ans;
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
int T = sc.nextInt();
for (int test_case = 1; test_case <= T; test_case++) {
N = sc.nextInt();
L = sc.nextInt();
scores = new int[N];
cals = new int[L];
for(int i = 0; i < N; i++) {
scores[i] = sc.nextInt();
cals[i] = sc.nextInt();
}
ans = 0;
makeBurger(0, 0, 0);
System.out.println("#" + test_case + " " + ans);
}
}
static void makeBurger(int idx, int sumScore, int sumCals) {
if (sumCals > L) { // 이미 칼로리가 제한을 넘어가게 되는 경우도 고려
return;
}
// 기저 조건
if (idx == N) { // 모든 재료를 고려 완료하였을 때
if ( ans < sumScore) {
ans = sumScore;
}
return;
}
// 재귀 부분
// 재료를 사용한 경우
makeBurger(idx + 1, sumScore + scores[idx], sumCals + cals[idx]);
// 재료를 사용하지 않은 경우
makeBurger(idx + 1, sumScore, sumCals);
}
}
N! = N * (N-1) * (N-2) * ... * 2 * 1
<중복이 없다는 가정 하에>
순열 구현1 (반복문)
public static void main(String[] args) {
nums = new int[] {1, 2, 3};
N = nums.length;
for(int i = 0; i < N; i++) {
for (int j = 0; j < N; j++) {
if( i != j ) {
for (int k = 0; k < N; k++) {
if ( k != i && k != j) {
System.out.println(nums[i] + " " + nums[j] + " " + nums[k]);
}
}
}
}
}
}
순열 구현2 (swap)
import java.util.Arrays;
public class 순열01_반복문2 {
static int[] nums;
static int N;
public static void main(String[] args) {
nums = new int[] {0, 1, 2};
N = nums.length;
perm(0);
}
// 바꿀 배열이 static으로 선언이 되어있는 경우, 인자로 보내지 않아도 된다.
static void swap(int a, int b) {
int tmp = nums[a];
nums[a] = nums[b];
nums[b] = tmp;
}
// idx = 현재 판단 위치
static void perm (int idx) {
if (idx == N) {
System.out.println(Arrays.toString(nums));
return;
}
for(int i = idx; i < N; i++) {
swap (i, idx);
perm (idx+1);
swap (i, idx); // 다음 swap을 위해서 원래대로 돌려줘야 한다.
}
}
}
순열 구현3 (방문체크)
import java.util.Arrays;
public class 순열01_방문체크 {
static int[] nums, result;
static int N;
static boolean[] visited;
public static void main(String[] args) {
nums = new int[] {0, 1, 2};
N = nums.length;
visited = new boolean[N];
result = new int[N];
perm(0);
}
// idx = 현재 판단 위치
static void perm (int idx) {
// 기저 조건
if(idx == N) {
System.out.println(Arrays.toString(result));
}
// 재귀 부분
for(int i = 0; i < N; i++) {
// 사용하지 않은 원소를 갖고 생성
if (visited[i]) continue; // 이미 사용하였다면 넘어간다.
result[idx] = nums[i];
visited[i] = true;
perm(idx+1);
visited[i] = false; // result는 덮어버리기 때문에 덮어쓰지 않아도 된다.
}
}
}
순열 구현4 (비트마스킹)
public class 순열01_비트마스킹 {
static int[] nums, result;
static int N;
public static void main(String[] args) {
nums = new int[] {0, 1, 2};
N = nums.length;
result = new int[N];
}
// idx = 현재 판단 위치
// visited = 사용한 원소를 기록하기 위한 정수
static void perm (int idx, int visited) {
// 기저 조건
if(idx == N) {
System.out.println(Arrays.toString(result));
}
// 재귀 부분
for(int i = 0; i < N; i++) {
// 사용하지 않은 원소를 갖고 생성
if ((visited & (1 << i)) != 0) continue; // 이미 사용하였다면 넘어간다.
result[idx] = nums[i];
perm(idx+1, visited | (1<<i));
// 인자로 받아서 넘기게 되는 경우, perm 안에서 연산도 가능하다. 다음 자리를 판단
}
}
}