Notice
Recent Posts
Recent Comments
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
Tags
- 알고리즘
- Git
- IntelliJ
- oracle
- 생성
- Vue
- GitHub
- 쿼리
- 넥사크로
- aws
- 방법
- 오라클
- 에러
- 시큐리티
- mybatis
- db
- kotlin
- Security
- Java
- JavaScript
- Eclipse
- Spring
- 스프링
- jquery
- 자바
- JPA
- 함수
- 코틀린
- error
- 프로그래머스
Archives
- Today
- Total
송민준의 개발노트
DFS(Depth-First Search) - 깊이 우선 탐색법 본문
● 개념
루트 노트에서 시작해서 다음 분기로 넘어가기 전에 해당 분기를 완벽하게 탐색하는 방법
- 한 방향으로 끝까지 가다가 더 이상 갈 수없게 되면 이전 갈림길로 돌아가서 다시 탐색을 진행하는 방법과 유사하다.
- 깊이 우선 탐색(DFS)이 너비 우선 탐색(BFS)보다 좀 더 간단하다.
- 단순 검색 속도 자체는 너비 우선 탐색(BFS)에 비해서 느리다.
● 특징
- 자기 자신은 호출하는 순환 알고리즘의 형태를 가지고 있다.
- 전위 순회(Pre Order)를 포함한 다른 형태(In Order, Post Order)의 트리 순회는 모두 DFS의 한 종류다.
- 특정 노드를 방문했는지 여부를 반드시 검사를 해야 한다. ( 무한루프 위험이 있음)
● 시간 복잡도
- DFS는 그래프의 모든 간선을 조회한다.(정점의 수 : N, 간선의 수 : E)
1) 인접 리스트로 표현된 그래프 : O(N+E)
2) 인접 행렬로 표현된 그래프 : O(N^2)
- 그래프 내에 적은 숫자의 간선만을 가지는 희소 그래프의 경우 인접 행렬보다 인접 리스트를 사용하는게 유리
● 접근 방법
- Stack
- Recursion(재귀)
class Graph {
class Node {
int data;
LinkedList<Node> adjacent;
boolean marked;
Node(int data) {
this.data = data;
this.marked = false;
adjacent = new LinkedList<Node>();
}
}
Node[] nodes;
Graph(int size) {
nodes = new Node[size];
for(int i =0; i<size; i++) {
nodes[i] = new Node(i);
}
}
void addEdge(int i1, int i2) {
Node n1 = nodes[i1];
Node n2 = nodes[i2];
}
if(!n1.adjacent.contains(n2)) {
n1.adjacent.add(n2);
}
if(!n2.adjacent.contains(n1)) {
n2.adjacent.add(n1);
}
void dfs() {
dfs(0);
}
void dfs(int index) {
Node root = nodes[index];
Stack<Node> stack = new Stack<Node>();
stack.push(root);
root.marked = true;
while(!stack.isEmpty()) {
Node r= stack.pop();
for(Node n : r.adjacent) {
if(n.marked == false) {
n.marked = true;
stack.push(n);
}
}
visit(r);
}
}
void bfs() {
bfs(0);
}
void bfs(int index) {
Node root = nodes[index];
Queue<Node> queue = new Queue<Node>();
queue.enqueue(root);
root.marked = true;
while(!queue.isEmpty()) {
Node r = queue.dequeue();
for(Node n : r.adjacent) {
if (n.marked == false) {
n.marked = true;
queue.enqueue(n);
}
}
visit(r);
}
}
void dfsR(Node r) {
if(r == null) return;
r.marked = true;
visit(r);
for(Node n : r.adjacent) {
if(n.marked == false) {
dfsR(n);
}
}
}
void dfsR(int index) {
Node r = nodes[index];
dfsR(r);
}
void dfsR() {
dfsR(0);
}
void visit(Node n) {
System.out.print(n.data + " ");
}
}
/*
0
/
1 -- 3 7
| / | \ /
| / | 5
2 -- 4 \
6 - 8
*/
public class Test {
public static void main(String[] args) {
Graph g = new Graph(9);
g.addEdge(0, 1);
g.addEdge(1, 2);
g.addEdge(1, 3);
g.addEdge(2, 4);
g.addEdge(2, 3);
g.addEdge(3, 4);
g.addEdge(3, 5);
g.addEdge(5, 6);
g.addEdge(5, 7);
g.addEdge(6, 8);
g.dfs();
g.bfs();
g.dfsR();
g.dfs(3);
g.bfs(3);
g.dfsR(3);
g.bfs();
}
}
'알고리즘' 카테고리의 다른 글
시간복잡도 Big-O 표기 (0) | 2020.02.03 |
---|---|
프로그래머스-level2-타겟넘버 (0) | 2019.12.13 |
프로그래머스-두 정수 사이의 합 (0) | 2019.10.29 |
프로그래머스-k번째수 (0) | 2019.10.28 |
프로그래머스-서울에서 김서방 찾기 (0) | 2019.10.28 |