[백준]1766: 문제집 - JAVA
[백준]1766: 문제집
1766번: 문제집
첫째 줄에 문제의 수 N(1 ≤ N ≤ 32,000)과 먼저 푸는 것이 좋은 문제에 대한 정보의 개수 M(1 ≤ M ≤ 100,000)이 주어진다. 둘째 줄부터 M개의 줄에 걸쳐 두 정수의 순서쌍 A,B가 빈칸을 사이에 두고 주
www.acmicpc.net
풀이
어제 위상정렬 공부한 기념으로 위상정렬 문제를 풀어보았다.
이 문제는 위상정렬 문제를 풀 줄 안다면 기본적인 개념을 이해하는데 어렵지 않다. 그러나 여기서 중요한 점은 문제 풀 순서의 조건중 3번인 가능한 쉬운 문제부터 풀어야 한다는 점이다.
예를들어서 예제입력을 보면, 4 -> 2, 3 -> 1순서대로 문제를 풀어야하고 3, 4의 진입간선이 모두 0이기 때문에 처음에 큐에 진입간선이 0인 값을 넣어주면 3, 4 순서대로 들어가게 된다.
그런데 이때 3을 꺼내고 3과 연결된 1의 진입간선을 다시 계산해주면 1의 진입간선이 0이되어 큐에 들어오게된다.
이때 큐의 상태는 4, 1이 될 것이다. 그렇다면 그다음으로 꺼내질 값은 4인데 문제 조건에 따라 4보다 1이 먼저 꺼내져야한다.(1이 4보다 더 쉬운 문제이기 때문이다.)
이를 어떻게 구현해야 할지 고민해 보다가 큐에 들어갈 순서를 조정해 주면 될 것이라는 생각이 들었다.
그래서 생각해낸게 '우선순위 큐'였다. 큐에 들어가고 난 후 큐 내부의 값을 우선순위에 따라 정렬시켜주면 앞선 상황에서 4보다 1이 먼저 꺼내질 것이고 이는 3번 조건을 항상 만족할것이다.
기본적인 위상정렬 알고리즘을 큐가 아닌 우선순위큐로 구현하면 되는 간단한 문제였다-!
채점을 해보니 메모리초과가 났다. String객체를 써서 그런 것 같아 Stringbuilder로 바꿔주니 해결되었다.
코드
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
46
47
48
49
50
51
52
53
|
import java.util.*;
public class Main {
static ArrayList<Integer>[] list;
static int[] indegree;
public static void main(String[] args) {
Scanner scan = new Scanner(System.in);
int n = scan.nextInt();
int m = scan.nextInt();
//연걸 리스트 생성
list = new ArrayList[n + 1];
for(int i = 0; i < n + 1; i++) {
list[i] = new ArrayList<>();
}
//연결 정보 입력, 진입 간선수 저장
indegree = new int[n + 1];
for (int i = 0; i < m; i++) {
int a = scan.nextInt();
int b = scan.nextInt();
indegree[b]++;
list[a].add(b);
}
System.out.println(topologicalSort());
}
public static String topologicalSort() {
PriorityQueue<Integer> q = new PriorityQueue<>();
for(int i = 1; i < indegree.length; i++) {
if(indegree[i] == 0) {
q.add(i);
}
}
StringBuilder sb = new StringBuilder();
while(!q.isEmpty()) {
int question = q.poll();
sb.append(question + " ");
for(int i = 0; i < list[question].size(); i++) {
int temp = list[question].get(i);
indegree[temp]--;
if(indegree[temp] == 0) q.add(temp);
}
}
return sb.toString();
}
}
|
cs |