일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
- 자료구조
- TensorFlow Lite
- DP문제
- 초소형머신러닝
- 코테 문제
- 삼성코테
- dfs
- DP
- 임베디드 딥러닝
- 코테
- 영상처리
- sort
- tinyml
- 삼성코딩테스트
- 다이나믹프로그래밍
- tflite
- 알고리즘
- 포스코 AI교육
- 포스코 ai 교육
- BFS
- 포스코 교육
- 삼성역량테스트
- 코딩테스트
- bfs문제
- 삼성역테
- 딥러닝
- dfs문제
- 컴퓨팅사고
- MCU 딥러닝
- 그리디
- Today
- Total
코딩뚠뚠
[백준문제풀이] 2150 Strongly Connected Component 본문
풀이일시 : 2020-08-29
문제 :
방향 그래프가 주어졌을 때, 그 그래프를 SCC들로 나누는 프로그램을 작성하시오.
방향 그래프의 SCC는 우선 정점의 최대 부분집합이며, 그 부분집합에 들어있는 서로 다른 임의의 두 정점 u, v에 대해서 u에서 v로 가는 경로와 v에서 u로 가는 경로가 모두 존재하는 경우를 말한다.
예를 들어 위와 같은 그림을 보자. 이 그래프에서 SCC들은 {a, b, e}, {c, d}, {f, g}, {h} 가 있다. 물론 h에서 h로 가는 간선이 없는 경우에도 {h}는 SCC를 이룬다.
입력:
첫째 줄에 두 정수 V(1 ≤ V ≤ 10,000), E(1 ≤ E ≤ 100,000)가 주어진다. 이는 그래프가 V개의 정점과 E개의 간선으로 이루어져 있다는 의미이다. 다음 E개의 줄에는 간선에 대한 정보를 나타내는 두 정수 A, B가 주어진다. 이는 A번 정점과 B번 정점이 연결되어 있다는 의미이다. 이때 방향은 A → B가 된다.
정점은 1부터 V까지 번호가 매겨져 있다.
ex)
7 9
1 4
4 5
5 1
1 6
6 7
2 7
7 3
3 7
7 2
출력:
첫째 줄에 SCC의 개수 K를 출력한다. 다음 K개의 줄에는 각 줄에 하나의 SCC에 속한 정점의 번호를 출력한다. 각 줄의 끝에는 -1을 출력하여 그 줄의 끝을 나타낸다. 각각의 SCC를 출력할 때 그 안에 속한 정점들은 오름차순으로 출력한다. 또한 여러 개의 SCC에 대해서는 그 안에 속해있는 가장 작은 정점의 정점 번호 순으로 출력한다.
ex)
3
1 4 5 -1
2 3 7 -1
6 -1
풀이 :
DFS를 이용하여 특정한 그래프 안에 포함되어있는 SCC를 출력하는 문제이다.
출력시에 가장 작은 정점의 번호 순으로 출력을 해야하므로 정렬을 수행해줘야한다.
#include <iostream>
#include <stack> //dfs위해
#include <vector>
#include <algorithm>
#define MAX 10001 //정점이 최대 만개이므로
using namespace std;
vector<int> a[MAX]; //인접노드들을 담을수있게
int id=0, d[MAX];
stack<int> s; //SCC판별시엔 스택이 쌓인다.
bool finished[MAX]; //처리가 된 노드 표시
vector<vector<int>> SCC;
int dfs(int x) { //항상 처음 방문하는 노드만 실행된다. 즉 n번수행
d[x] = ++id; //d[x]가 0이 아님만 나타냄
s.push(x); //들어온숫자를 스택에 넣는다.
int result= d[x];
for (int i = 0; i < a[x].size(); i++) { //연결된거 돌아가면서
int y = a[x][i]; //y초기화
if (d[y] == 0) //아직 방문하지않은 이웃노드인경우
result = min(result, dfs(y)); //재귀로 dfs한 후 더 작은값이 result
else if (!finished[y]) //방문은했지만 아직 처리중인 이웃노드이면
//아직 처리중이라는 것은 즉 자신의 부모일 것이라는것 자신과 연결되었다는것
result = min(result, d[y]); //해당 이웃노드의 d배열값을 넣어준다
}
//모든 인접노드를 확인한 다음에 돌아왔을때
//부모노드가 자기 자신인 경우 SCC를 형성한다.
if (result == d[x]) { //즉 이웃노드를 전부 방문한 이후에도 자기 자신이
//min 인 result 라면 내가 특정한 scc에서 부모역할을 수행하는 것이였구나 라는 것을 깨달은것
vector<int> scc;
while (1) { //자기 자신이 나올때까지 스택에서 꺼내면 된다.
int t = s.top();
s.pop();
scc.push_back(t); //작은 scc에 t를 넣어
finished[t] = true; //빠져나온 값은 finished 를 true로만듦
if (t == x) //t가 부모면 멈춘다.
break;
}
sort(scc.begin(), scc.end());
SCC.push_back(scc);
}
return result;
}
int main() {
int v, e;
scanf_s("%d %d", &v, &e);
for (int i = 0; i < e; i++) { //간선 수만큼 반복
int x, y;
scanf_s("%d %d", &x, &y);
a[x].push_back(y); //뭐와 뭐가 연결되어있는지
}
for (int i = 1; i < v; i++) { //1에서부터 정점갯수 전까지 반복
if (d[i] == 0) //방문체크 안됐으면(근데 아예 떨어져있는 경우 아니면 한면에 다 훑는다.)
dfs(i); //dfs 실행
}
printf("%d\n", SCC.size()); //SCC사이즈 출력하고
sort(SCC.begin(), SCC.end()); //작은 숫자먼저로 정렬시킨다.
for (int i = 0; i < SCC.size(); i++) { //사이즈 만큼 반복
for (int j = 0; j < SCC[i].size(); j++) { //SCC구성하는거 프린트
printf("%d ", SCC[i][j]); //vector<vector<int>> SCC 로 선언했다.
}
printf("-1\n");
}
return 0;
}
'알고리즘 문제풀이 > 백준문제풀이' 카테고리의 다른 글
[백준문제풀이] 3977 축구전술 (0) | 2020.12.30 |
---|---|
[백준문제풀이] 4196 도미노 (0) | 2020.12.30 |
[백준문제풀이] 1671 상어의 저녁식사 (0) | 2020.12.30 |
[백준문제풀이] 11377 열혈강호3 (0) | 2020.12.30 |
[백준문제풀이] 11376 열혈강호2 (0) | 2020.12.30 |