[백준 1766] 문제집 (C#, C++) - soo:bak
작성일 :
문제 링크
설명
N개의 문제를 풀어야 하는데, 일부 문제는 선행 관계가 있어서 특정 문제를 먼저 풀어야 합니다. 선행 관계를 지키면서 가능하면 쉬운 번호부터 푸는 순서를 구하는 문제입니다.
접근법
먼저 “지금 풀 수 있는 문제”를 정의합니다. 선행 문제가 없거나, 선행 문제를 모두 푼 문제가 지금 풀 수 있는 문제입니다.
다음으로 매 순간 지금 풀 수 있는 문제 중 번호가 가장 작은 것을 고릅니다. 최소 힙에 지금 풀 수 있는 문제들을 넣어두면, 항상 가장 작은 번호를 빠르게 꺼낼 수 있습니다.
이후 문제를 풀 때마다 그 문제를 선행 조건으로 가지던 문제들을 확인합니다. 선행 조건이 모두 해결되었으면 해당 문제도 힙에 추가합니다.
시간 복잡도는 O((N + M) log N)입니다.
Code
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
using System;
using System.Collections.Generic;
using System.Text;
namespace Solution {
class Program {
static void Main(string[] args) {
var first = Array.ConvertAll(Console.ReadLine()!.Split(), int.Parse);
var n = first[0];
var m = first[1];
var adj = new List<int>[n + 1];
for (var i = 1; i <= n; i++)
adj[i] = new List<int>();
var indeg = new int[n + 1];
for (var i = 0; i < m; i++) {
var line = Array.ConvertAll(Console.ReadLine()!.Split(), int.Parse);
var a = line[0];
var b = line[1];
adj[a].Add(b);
indeg[b]++;
}
var pq = new PriorityQueue<int, int>();
for (var i = 1; i <= n; i++) {
if (indeg[i] == 0)
pq.Enqueue(i, i);
}
var sb = new StringBuilder();
while (pq.Count > 0) {
pq.TryDequeue(out var cur, out _);
sb.Append(cur).Append(' ');
foreach (var nxt in adj[cur]) {
indeg[nxt]--;
if (indeg[nxt] == 0)
pq.Enqueue(nxt, nxt);
}
}
Console.WriteLine(sb.ToString().TrimEnd());
}
}
}
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
#include <bits/stdc++.h>
using namespace std;
typedef vector<int> vi;
typedef vector<vi> vvi;
int main() {
ios::sync_with_stdio(false);
cin.tie(nullptr);
int n, m; cin >> n >> m;
vvi adj(n + 1);
vi indeg(n + 1, 0);
for (int i = 0; i < m; i++) {
int a, b; cin >> a >> b;
adj[a].push_back(b);
indeg[b]++;
}
priority_queue<int, vi, greater<int>> pq;
for (int i = 1; i <= n; i++) {
if (indeg[i] == 0)
pq.push(i);
}
ostringstream out;
while (!pq.empty()) {
int cur = pq.top(); pq.pop();
out << cur << ' ';
for (int nxt : adj[cur]) {
if (--indeg[nxt] == 0)
pq.push(nxt);
}
}
string res = out.str();
if (!res.empty() && res.back() == ' ')
res.pop_back();
cout << res << "\n";
return 0;
}