zoukankan      html  css  js  c++  java
  • Summary: Depth-first Search(DFS)

    There are generally two methods to write DFS algorithm, one is using recursion, another one is using stack. (reference from Wiki Pedia)

    Pseudocode for both methods:

    A recursive implementation of DFS:

    1 procedure DFS(G,v):
    2       label v as discovered
    3       for all edges from v to w in G.adjacentEdges(v) do
    4           if vertex w is not labeled as discovered then
    5               recursively call DFS(G,w)

    A non-recursive implementation of DFS(using stack):

    1   procedure DFS-iterative(G,v):
    2       let S be a stack
    3       S.push(v)
    4       while S is not empty
    5             v ← S.pop() 
    6             if v is not labeled as discovered:
    7                 label v as discovered
    8                 for all edges from v to w in G.adjacentEdges(v) do
    9                     S.push(w)

    The non-recursive implementation is similar to breadth-first search but differs from it in two ways: it uses a stack instead of a queue, and it delays checking whether a vertex has been discovered until the vertex is popped from the stack rather than making this check before pushing the vertex.

    Here's another good explenation: http://www.algolist.net/Algorithms/Graph/Undirected/Depth-first_search

    Depth-first search, or DFS, is a way to traverse the graph. Initially it allows visiting vertices of the graph only, but there are hundreds of algorithms for graphs, which are based on DFS. Therefore, understanding the principles of depth-first search is quite important to move ahead into the graph theory. The principle of the algorithm is quite simple: to go forward (in depth) while there is such possibility, otherwise to backtrack.

    Algorithm

    In DFS, each vertex has three possible colors representing its state:

    white: vertex is unvisited;

    gray: vertex is in progress;

    black: DFS has finished processing the vertex.

    NB. For most algorithms boolean classification unvisited / visited is quite enough, but we show general case here.

    Initially all vertices are white (unvisited). DFS starts in arbitrary vertex and runs as follows:

    1. Mark vertex u as gray (visited).
    2. For each edge (u, v), where u is white, run depth-first search for u recursively.
    3. Mark vertex u as black and backtrack to the parent.

    Example. Traverse a graph shown below, using DFS. Start from a vertex with number 1.

    depth-first search example, pic. 1 Source graph.
    Mark a vertex 1 as gray.
    There is an edge (1, 4) and a vertex 4 is unvisited. Go there.
    Mark the vertex 4 as gray.
    There is an edge (4, 2) and vertex a 2 is unvisited. Go there.
    Mark the vertex 2 as gray.
    There is an edge (2, 5) and a vertex 5 is unvisited. Go there.
    Mark the vertex 5 as gray.
    There is an edge (5, 3) and a vertex is unvisited. Go there.
    Mark the vertex 3 as gray.
    There are no ways to go from the vertex 3. Mark it as black and backtrack to the vertex 5.
    There is an edge (5, 4), but the vertex 4 is gray.
    There are no ways to go from the vertex 5. Mark it as black and backtrack to the vertex 2.
    There are no more edges, adjacent to vertex 2. Mark it as black and backtrack to the vertex 4.
    There is an edge (4, 5), but the vertex 5 is black.
    There are no more edges, adjacent to the vertex 4. Mark it as black and backtrack to the vertex 1.
    There are no more edges, adjacent to the vertex 1. Mark it as black. DFS is over.


    Complexity analysis
    If a graph is disconnected, DFS won't visit all of its vertices. For details, see 
    finding connected components algorithm.As you can see from the example, DFS doesn't go through all edges. The vertices and edges, which depth-first search has visited is a tree. This tree contains all vertices of the graph (if it is connected) and is called graph spanning tree. This tree exactly corresponds to the recursive calls of DFS.

    Assume that graph is connected. Depth-first search visits every vertex in the graph and checks every edge its edge. Therefore, DFS complexity is O(V + E). As it was mentioned before, if an adjacency matrix is used for a graph representation, then all edges, adjacent to a vertex can't be found efficiently, that results in O(V2)complexity. You can find strong proof of the DFS complexity issues in [1].

    Java Implementation

     1 public class Graph {
     2  3      
     4 enum VertexState {
     5           White, Gray, Black 
     6       }
     7      
     8       public void DFS()
     9       {
    10             VertexState state[] = new VertexState[vertexCount];
    11             for (int i = 0; i < vertexCount; i++)
    12                   state[i] = VertexState.White;
    13             runDFS(0, state);
    14       }
    15      
    16       public void runDFS(int u, VertexState[] state)
    17       {
    18             state[u] = VertexState.Gray;
    19             for (int v = 0; v < vertexCount; v++)
    20                   if (isEdge(u, v) && state[v] == VertexState.White)
    21                         runDFS(v, state);
    22             state[u] = VertexState.Black;
    23       }
    24 }
  • 相关阅读:
    http://knockoutjs.com/工作杂记
    wp开发杂记
    [转载]jQuery 顺便学习下CSS选择器 奇偶匹配nthchild(even)
    [转载]各种图像处理类库的比较及选择(The Comparison of Image Processing Libraries)
    [转载]多维数组与Json格式的转化
    [转载]OpenCv,EmguCv及.net之间的互动(The Interaction of OpenCv, EmguCv AND .net)
    [转载].net中捕获摄像头视频的方式及对比(How to Capture Camera Video via .Net)
    paip.杀不死进程的原因僵尸进程的解决.txt
    paip.c3p0 数据库连接池 NullPointerException 的解决...
    paip..禁用mmseg 的默认词库. . 仅仅使用自定义词库from数据库.
  • 原文地址:https://www.cnblogs.com/EdwardLiu/p/3781128.html
Copyright © 2011-2022 走看看