[백준 2606번] 바이러스

2017. 5. 16. 01:33알고리즘/백준

반응형





풀이


문제를 알고리즘 유형에 맞게 바꾸어 해석해보면 다음과 같다.


1. 출발지점은 무조건 1번부터 시작한다. (1번 컴퓨터는 무조건 감염 되어있다.)

2. 출발지점부터 그다음 갈 수 있는 지점까지를 카운트 한다. (1번부터 연결되어 있는 컴퓨터를 모두 찾아야 한다.)


이렇게 보면 출발지점이 고정되어 있는 일반적인 BFS 문제와 같다.


만약 2하고 4가 연결되어 있다면 2번지점에 있을때 4번으로 갈 수 있고, 4번지점에 있을때 2번으로 갈 수 있다. (양방향 간선이다.)

2차원 배열을 만들어서 연결되어 있다면 1, 없다면 0을 넣어두고 array[컴퓨터1][컴퓨터2] == 1 일때만 큐에 삽입할 수 있게한다.

이 때 양방향으로 갈 수 있으므로 array[컴퓨터2][컴퓨터1] == 1 일때도 생각해 주어야한다.



순서대로 해석하면


1. 1번이 시작지점이므로 1번에서 갈수 있는 모든 컴퓨터를 큐에 삽입한다.

2. 큐에서 하나를 꺼낸 숫자가 X라고 가정하였을 때 방문배열을 체크하고 X에서 갈 수 있는 모든 컴퓨터를 큐에 삽입한다.

3. 방문 배열에 체크된 갯수를 센다.





소스 코드


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
54
55
56
57
58
#include <iostream>
using namespace std;
 
int com, con, cnt=0;
int network[101][101];
int que[10001];
int head = 0, tail = 0;
int visit[101];
 
void push(int a) {
    que[tail++= a;
}
 
int pop() {
    return que[head++];
}
 
void bfs() {
    while (head != tail) {
        int a = pop();
        visit[a] = 1;
        for (int i = 1; i <= com; i++) {
            if (network[a][i] == 1) {
                push(i);
                network[a][i] = 2;
                network[i][a] = 2;
            }
        }
    }
}
 
int main() {
    cin >> com >> con;
    for (int i = 1; i <= com; i++) {
        for (int j = 1; j <= com; j++) {
            network[i][j] = 0;
        }
        visit[i] = 0;
    }
    for (int i = 0; i < con; i++) {
        int a, b;
        cin >> a >> b;
        network[a][b] = 1;
        network[b][a] = 1;
    }
 
    push(1);
    bfs();
 
    for (int i = 1; i <= com; i++) {
        if (visit[i]) {
            cnt++;
        }
    }
 
    cout << cnt-1 << '\n';
    return 0;
}
cs





DP란


설명[편집]

동적 계획법의 원리는 매우 간단하다. 일반적으로 주어진 문제를 풀기 위해서, 문제를 여러 개의 하위 문제(subproblem)로 나누어 푼 다음, 그것을 결합하여 최종적인 목적에 도달하는 것이다. 각 하위 문제의 해결을 계산한 뒤, 그 해결책을 저장하여 후에 같은 하위 문제가 나왔을 경우 그것을 간단하게 해결할 수 있다. 이러한 방법으로 동적 계획법은 계산 횟수를 줄일 수 있다. 특히 이 방법은 하위 문제의 수가 기하급수적으로 증가할 때 유용하다.

동적 계획 알고리즘은 최단 경로 문제, 행렬의 제곱 문제 등의 최적화에 사용된다. 이것은 동적 계획법은 문제를 해결하기 위한 모든 방법을 검토하고, 그 중에 최적의 풀이법을 찾아내기 때문이다. 이에 우리는 동적 계획법을 모든 방법을 일일이 검토하여 그 중 최적해를 찾아내는 주먹구구식 방법이라고 생각할 수 있다. 그러나 문제가 가능한 모든 방법을 충분히 빠른 속도로 처리할 수 있는 경우, 동적 계획법은 최적의 해법이라고 말할 수 있다.

때로는 단순한 재귀함수에 저장 수열(이전의 데이터를 모두 입력하는 수열)을 대입하는 것 만으로도 최적해를 구할 수 있는 동적 알고리즘을 찾을 수 있다. 그러나 대다수의 문제는 이보다 훨씬 더 복잡한 프로그래밍을 요구한다. 그 중에 일부는 여러 개의 매개 변수를 이용하여 재귀 함수를 작성해야 하는 것도 있고, 아예 이러한 방법으로 동적 알고리즘을 짤 수 없는 문제 또한 존재한다. 이러한 퍼즐로는 대표적으로 Egg Dropping Puzzle이 있다.

그리디 알고리즘과의 비교[편집]

동적 계획법은 위에서 설명했듯이, 주먹구구식의 방법이라는 단점이 있다. 이러한 단점을 극복하기 위하여, 동적 계획법 대신 그리디 알고리즘 이 등장했다. 그리디 알고리즘은 항상 최적해를 구해주지는 않지만, 다행히 MST(최소 비용 나무 문제) 등의 여러 문제에서 그리디 알고리즘이 최적해를 구할 수 있음이 이미 입증되었다.

그리디 알고리즘과 동적 계획법을 비교하자. 우리가 차량 정체 구간에서 A라는 지점에서 B라는 지점까지 가능한 빨리 이동하는 경로를 찾고 싶다고 하자. 이 문제에서 동적 계획법을 사용한다면, 우리가 갈 수 있는 모든 상황과 교통 정체를 전부 감안하여 최적의 경로를 찾아낸다. 반면 그리디 알고리즘은 전체적인 상황을 고려하지 않고, 순간순간 교차로가 보일 때마다 가장 빠른 경로를 검색하여 찾아줄 것이다.

물론 동적 계획법으로 경로를 검색하는 동안 우리가 운전을 잠깐 쉬어야 하듯이, 우리는 동적 계획법을 사용하면 약간의 시간이 걸린다는 단점이 있다. 그러나 이렇게 얻어낸 경로는 (교통 환경이 변하지 않았다는 가정 하에) 우리가 갈 수 있는 가장 빠른 길이 된다고 장담할 수 있다. 반면 그리디 알고리즘은 즉효성이 있는 대신, 항상 최적의 경로를 찾아주지는 않는다. 각 구간마다 최적의 경로를 찾는다고 해도 그것이 전체적으로 최적의 경로가 되지는 않기 때문이다. 즉, 동적 계획법은 그리디 알고리즘에 비해 시간적으로는 효율적이지 못할 수는 있어도, 그 결과에 대해서는 효율적인 값을 구할 수가 있다.



[출처 : https://ko.wikipedia.org/wiki/%EB%8F%99%EC%A0%81_%EA%B3%84%ED%9A%8D%EB%B2%95]


반응형

'알고리즘 > 백준' 카테고리의 다른 글

[백준 1463번] 1로 만들기  (0) 2017.05.17
[백준 1966번] 프린터 큐  (0) 2017.05.16
[백준 2193번]이친수  (0) 2017.05.15
[백준 11726번] 2xn 타일링  (3) 2017.05.15
[백준 2579번] 계단 오르기  (9) 2017.05.14