Min:D's Devlog

[SWEA][모의 SW 역량테스트][C++] 4012 요리사 본문

알고리즘/SWEA

[SWEA][모의 SW 역량테스트][C++] 4012 요리사

Min:D 2020. 7. 9. 22:46

문제

SWEA 모의 SW 역량테스트 - 4012 요리사

 

SW Expert Academy

SW 프로그래밍 역량 강화에 도움이 되는 다양한 학습 컨텐츠를 확인하세요!

swexpertacademy.com

 


문제 풀이

접근 방식

이 문제는 N(짝수) 개의 식재료를 반으로 나눠 두 개의 요리를 할 때,

두 요리의 총 시너지의 차가 최소가 되도록 조합하는 문제이다.

조합을 간단하게 구현하기 위해 next_permutation을 사용해주었다.

 

next_permutation에 인자로 넣을 벡터 idx는 N개의 0으로 초기화하고,

N/2개의 1을 뒤쪽부터 넣어주었다.(정렬할 필요가 없도록!)

do ~ while문 내에서는 0인 경우와 1인 경우로 나눠서 인덱스를 벡터에 저장하고,

그 인덱스들의 조합을 통해 각 요리의 총 시너지를 구해주었다.

(i == j인 경우, 시너지 값이 0이기 때문에 예외 처리 없이 더해줘도 된다.)

abs를 사용하여 두 시너지의 값의 차를 구하고, 이를 활용하여 최솟값을 찾아주었다.

 


풀이 코드 - C++

#include<iostream>
#include <vector>
#include <algorithm>
 
using namespace std;
 
int main(int argc, char** argv)
{
    int test_case;
    int T, N;
 
    // freopen("input.txt", "r", stdin);
    cin >> T;
 
    for (test_case = 1; test_case <= T; ++test_case) {
        cin >> N;
        vector<vector<int>> S(N, vector<int>(N));
        for (int i = 0; i < N; i++)  {
            for (int j = 0; j < N; j++)  {
                cin >> S[i][j];
            }
        }
        vector<bool> idx(N);
        for (int i = 0; i < N / 2; i++) {
            idx[N - i - 1] = 1;
        }
        int MIN = 987654321;
        do {
            int sum1 = 0;
            int sum2 = 0;
            vector<int> num1;
            vector<int> num2;
            for (int i = 0; i < N; i++)  {
                if (idx[i] == 1) {
                    num1.push_back(i);
                }
                else {
                    num2.push_back(i);
                }
            }
            for (int i = 0; i < N/2; i++) {
                for (int j = 0; j < N/2; j++) {
                    sum1 += S[num1[i]][num1[j]];
                    sum2 += S[num2[i]][num2[j]];
                }
            }
            int result = abs(sum1 - sum2);
            if (MIN > result)
                MIN = result;
        } while (next_permutation(idx.begin(), idx.end()));
 
        cout << "#" << test_case << " " << MIN << endl;
    }
    return 0;
}

실행 결과

 

Comments