Problem Solving/BOJ

[백준 / BOJ] C++ 4792 레드 블루 스패닝 트리

nageune 2023. 3. 30. 23:25
728x90
반응형

4792번: 레드 블루 스패닝 트리

 

문제

https://www.acmicpc.net/problem/4792

 

4792번: 레드 블루 스패닝 트리

무방향, 무가중치, 연결 그래프가 주어진다. 그래프의 각 간선은 빨간색 또는 파란색으로 색칠되어져 있다. 이 그래프의 스패닝 트리 중 파란색 간선이 정확히 k개인 것이 있는지 없는지 알아내

www.acmicpc.net

 

 

풀이

파란색 간선의 가중치를 1, 빨간색 간선의 가중치를 0으로 해서 간선을 추가한다. 간선을 가중치를 기준으로 오름차순 정렬하여 파란색 간선의 수를 최소로 한 스패닝 트리의 가중치(파란색 간선의 수)를 minBlue, 내림차순 정렬하여 최대로 한 스패닝 트리의 가중치를 maxBlue로 저장한다. minBlue ≤ K ≤ maxBlue를 만족하면 1, 만족하지 않으면 0을 출력한다.

 

 

코드

#include <bits/stdc++.h>
using namespace std;

vector<int> p;
vector<pair<int, pair<int, int>>> edge;

int find(int x) {
  if (p[x] != x)
    p[x] = find(p[x]);
  return p[x];
}

void merge(int x, int y) {
  x = find(x);
  y = find(y);
  if (x < y)
    p[y] = x;
  else
    p[x] = y;
}

int kruskal() {
  iota(p.begin(), p.end(), 0);
  int cnt = 0;
  for (auto [weight, path] : edge)
    if (find(path.first) != find(path.second)) {
      cnt += weight;
      merge(path.first, path.second);
    }
  return cnt;
}

int main() {
  ios::sync_with_stdio(false);
  cin.tie(NULL);
  while (1) {
    int n, m, k;
    cin >> n >> m >> k;
    if (n + m + k == 0)
      break;
    p.resize(n + 1);
    edge.clear();
    while (m--) {
      char c;
      int f, t;
      cin >> c >> f >> t;
      if (c == 'B')
        edge.push_back({1, {f, t}});
      else
        edge.push_back({0, {f, t}});
    }
    sort(edge.begin(), edge.end());
    int minBlue = kruskal();
    sort(edge.begin(), edge.end(), greater<>());
    int maxBlue = kruskal();
    if (minBlue <= k && k <= maxBlue)
      cout << "1\n";
    else
      cout << "0\n";
  }
  return 0;
}

 

728x90
반응형