ABOUT ME

-

Today
-
Yesterday
-
Total
-
  • 백준 14889 (스타트와 링크) c++
    백준 문제 2022. 1. 10. 18:48
    728x90

    문제

     

    14889번: 스타트와 링크

    예제 2의 경우에 (1, 3, 6), (2, 4, 5)로 팀을 나누면 되고, 예제 3의 경우에는 (1, 2, 4, 5), (3, 6, 7, 8)로 팀을 나누면 된다.

    www.acmicpc.net

    이 문제의 관건은 팀을 2개로 나누는 여러가지 조합에대해서 처리해주어야하는 것입니다.

    여기서 포인트는 중복이 아니라 조합이라는 것입니다.

    1, 2, 3, 4 중에서 start 팀 = (1, 2)  link 팀 =(3, 4) 과 start 팀 = (3, 4), link 팀 =(1, 2) 는 2가지가 아니라 1가지로

    취급하는 것입니다.

    int n;
    int arr[21][21];
    bool check[21];
    int ans = INT_MAX;
    void dfs(int cur, int next);
    int main() {
    	ios_base::sync_with_stdio(false), cin.tie(NULL), cout.tie(NULL);
    	cin >> n;
    	for (int i = 1; i <= n; i++) {
    		for (int j = 1; j <= n; j++) {
    			cin >> arr[i][j];
    		}
    	}

    arr은 배열을 받고, 변수 ans는 두 팀의 능력치의 차의 최솟값을 넣을 것입니다. 

    check 배열은 dfs를 실행할 때 방문한곳을 체크하기 위함입니다.

    void dfs(int cur, int next) {
    	if (cur == n / 2) {
    		int start = 0;
    		int link = 0;
    
    		for (int i = 1; i <= n; i++) {
    			for (int j = 1; j <= n; j++) {
    				if (check[i] == true && check[j] == true)start += arr[i][j];
    				if (check[i] == false && check[j] == false)link += arr[i][j];
    			}
    		}
    		int temp = abs(start - link);
    		if (ans > temp)ans = temp;
    
    		return;
    	}
    
    	for (int i = next; i < n; i++) {
    		check[i] = true;
    		dfs(cur + 1, i + 1);
    		check[i] = false;
    	}
    }

    main 함수에서 dfs(0, 1)을 호출합니다. 변수 cur는 지금까지 뽑은 팀원의 수 입니다. 

    변수 next는 현재 check 배열에서 뽑을 인덱스 위치 입니다. 

    4개의 팀원이 있으면 2명이 팀이 되면 자동적으로 팀이 구성되므로 cur이 n/2 가 되면 본격적으로 계산을 진행합니다.

    for (int i = next; i < n; i++) {
    	check[i] = true;
    	dfs(cur + 1, i + 1);
    	check[i] = false;
    }

    dfs 함수 마지막에 있는 부분입니다. 

    여기서 for문을 i<n까지로 설정했는데 이렇게 설정하게 되면

    1, 2, 3, 4 중에서

    {1, 2}

    {1, 3}

    {2, 3}

    만을 탐색하게 됩니다.

    이렇게 되면 

    start          /       link

    {1, 2}                 {3, 4}

    {1, 3}                 {2, 4}

    {2, 3}                 {1, 4}

    가 자동적으로 완성하게 됩니다. 

     

    반면에 for문을 i<=n 까지로 설정하면

    1, 2, 3, 4 중에

    {1, 2}

    {1, 3}

    {1, 4}

    {2, 3}

    {2, 4}

    {3, 4}

    까지 모두 탐색하게 됩니다.

    이렇게 되면

    start            /     link

    {1, 2}                 {3, 4}

    {1, 3}                 {2, 4}

    {1, 4}                 {2, 3}

    {2, 3}                 {2, 4}

    {2, 4}                 {1, 3}

    {3, 4}                 {1, 2}

    중복이 발생하게 됩니다.


    전체 코드입니다.

    #include<iostream>
    #include<algorithm>
    #include<climits>
    #include<string>
    #include<vector>
    #include<queue>
    #include<stack>
    #include<deque>
    #include<cmath>
    #include<time.h>
    #include<cstring>
    #include<cmath>
    using namespace std;
    using ll = long long;
    using ull = unsigned long long;
    int n;
    int arr[21][21];
    bool check[21];
    int ans = INT_MAX;
    void dfs(int cur, int next);
    int main() {
    	ios_base::sync_with_stdio(false), cin.tie(NULL), cout.tie(NULL);
    	cin >> n;
    	for (int i = 1; i <= n; i++) {
    		for (int j = 1; j <= n; j++) {
    			cin >> arr[i][j];
    		}
    	}
    	dfs(0, 1);
    
    	cout << ans;
    	return 0;
    }
    void dfs(int cur, int next) {
    	if (cur == n / 2) {
    		int start = 0;
    		int link = 0;
    
    		for (int i = 1; i <= n; i++) {
    			for (int j = 1; j <= n; j++) {
    				if (check[i] == true && check[j] == true)start += arr[i][j];
    				if (check[i] == false && check[j] == false)link += arr[i][j];
    			}
    		}
    		int temp = abs(start - link);
    		if (ans > temp)ans = temp;
    
    		return;
    	}
    
    	for (int i = next; i < n; i++) {
    		check[i] = true;
    		dfs(cur + 1, i + 1);
    		check[i] = false;
    	}
    }

    '백준 문제' 카테고리의 다른 글

    백준 2026 (소풍) c++  (0) 2022.01.11
    백준 2239 (스도쿠) c++  (0) 2022.01.11
    백준 1182 (부분수열의 합) c++  (0) 2022.01.09
    백준 9663 (N-Queen) c++  (0) 2022.01.06
    백준 14888 (연산자 끼워넣기) C++  (0) 2022.01.05
Designed by Tistory.