PS (C, C++)
[백준/C++] 2637 장난감 조립
최연재
2024. 9. 24. 21:56
https://www.acmicpc.net/problem/2637
코드
#include <iostream>
#include <vector>
#include <queue>
using namespace std;
#define FASTIO ios::sync_with_stdio(false);cin.tie(NULL);
typedef pair<int, int> pi;
int n,m;
int main() {
FASTIO;
cin >> n >> m;
vector<int>depth(n + 1, 0);
vector<vector<pi>> p(n+1);
vector<vector<int>> cnt(n + 1, vector<int>(n + 1, 0));
for (int i = 0,x, y, k; i < m; i++) {
cin >> x >> y >> k;
depth[x]++;
p[y].push_back({ x,k });
}
queue<int> q;
for (int i = 1; i <= n; i++)
if (depth[i] == 0) {
q.push(i);
cnt[i][i] = 1;
}
while (!q.empty()) {
int now = q.front();
q.pop();
for (int i = 0; i < p[now].size(); i++) {
int next = p[now][i].first;
int need = p[now][i].second;
for (int j = 1; j <=n; j++) {
if (cnt[now][j] != 0)
cnt[next][j] += cnt[now][j] * need;
}
depth[next]--;
if (depth[next] == 0)
q.push(next);
}
}
for (int i = 1; i < n; i++)
if (cnt[n][i] > 0)
cout << i << " " << cnt[n][i] << "\n";
return 0;
}
설명
필요한 변수를 입력받습니다. 부품들 간의 관계 X, Y, K는 X를 만드는 데 Y가 K개 필요하다는 의미이므로 depth[X]를 1 증가시켜 필요한 부품의 종류를 세어줍니다. 그리고 P[Y].push_back({X, K})로 Y는 추후 X를 만들기 위해 K개 필요함을 기록합니다.
depth[i] == 0 이라면 해당 부품을 만들기 위해서 필요한 다른 부품이 없다는 의미입니다. 따라서 cnt[i][i]를 1로 저장해주고 위상정렬을 위한 큐에 i를 넣어줍니다. 이때 cnt[i][j]는 i 부품을 만들기 위해 필요한 j 부품의 개수입니다.
이제 위상정렬을 진행합니다. p[now]에는 Y가 k개 필요한 부품 X의 정보가 들어있습니다. 따라서 매번 cnt[now][j]가 0이 아닌 값에 대해서 cnt[next][j] += cnt[now][j] * k 로 갱신합니다. 이렇게 하면 next 부품을 만들기 위해 필요한 선수 부품 한 종류에 대한 처리가 끝났으니 depth를 1 감소시켜줍니다. 0이 되었다면 이제 해당 부품 번호를 큐에 넣습니다.
위상정렬 이후 cnt[n][i]가 양수인 경우에만 주어진 형식대로 출력하고 프로그램을 종료합니다.
느낀 점
부품 사이 선수관계가 있기 때문에 위상정렬로 문제를 해결했습니다! 처음에 정보 관리를 잘못 생각해서 코드를 작성하는 과정이 오래 걸렸습니다.