rgl 0.2.2 → 0.2.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/ChangeLog +75 -2
- data/README +52 -28
- data/Rakefile +3 -3
- data/TAGS +242 -198
- data/examples/debgraph.rb +118 -0
- data/examples/examples.rb +5 -3
- data/examples/graph.dot +731 -17
- data/examples/insel.rb +141 -0
- data/lib/rgl/adjacency.rb +172 -139
- data/lib/rgl/base.rb +247 -251
- data/lib/rgl/connected_components.rb +125 -112
- data/lib/rgl/dot.rb +54 -46
- data/lib/rgl/graphxml.rb +48 -37
- data/lib/rgl/implicit.rb +159 -136
- data/lib/rgl/mutable.rb +69 -48
- data/lib/rgl/rdot.rb +268 -205
- data/lib/rgl/topsort.rb +63 -52
- data/lib/rgl/transitiv_closure.rb +40 -28
- data/lib/rgl/traversal.rb +300 -247
- data/tests/TestDirectedGraph.rb +22 -2
- data/tests/TestUnDirectedGraph.rb +4 -0
- metadata +7 -7
- data/Makefile +0 -72
- data/examples/graph.png +0 -0
data/lib/rgl/topsort.rb
CHANGED
@@ -1,61 +1,72 @@
|
|
1
|
+
# topsort.rb
|
2
|
+
|
1
3
|
require 'rgl/traversal'
|
2
4
|
|
3
5
|
module RGL
|
6
|
+
|
4
7
|
# Topological Sort Iterator
|
5
8
|
#
|
6
|
-
# The topological sort algorithm creates a linear ordering
|
7
|
-
#
|
8
|
-
#
|
9
|
-
# be a directed acyclic graph (DAG).
|
9
|
+
# The topological sort algorithm creates a linear ordering of the vertices
|
10
|
+
# such that if edge (u,v) appears in the graph, then u comes before v in
|
11
|
+
# the ordering. The graph must be a directed acyclic graph (DAG).
|
10
12
|
#
|
11
|
-
# The iterator can also be applied to undirected graph or a DG graph
|
12
|
-
# contains a cycle.
|
13
|
-
# implementation of acyclic? uses this fact.
|
13
|
+
# The iterator can also be applied to undirected graph or to a DG graph
|
14
|
+
# which contains a cycle. In this case, the Iterator does not reach all
|
15
|
+
# vertices. The implementation of acyclic? uses this fact.
|
16
|
+
|
14
17
|
class TopsortIterator
|
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
|
-
|
18
|
+
|
19
|
+
include GraphIterator
|
20
|
+
|
21
|
+
def initialize (g)
|
22
|
+
super(g)
|
23
|
+
set_to_begin
|
24
|
+
end
|
25
|
+
|
26
|
+
def set_to_begin # :nodoc:
|
27
|
+
@waiting = Array.new
|
28
|
+
@inDegrees = Hash.new(0)
|
29
|
+
|
30
|
+
graph.each_vertex do |u|
|
31
|
+
@inDegrees[u] = 0 unless @inDegrees.has_key?(u)
|
32
|
+
graph.each_adjacent(u) do |v|
|
33
|
+
@inDegrees[v] += 1
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
@inDegrees.each_pair do |v, indegree|
|
38
|
+
@waiting.push(v) if indegree.zero?
|
39
|
+
end
|
40
|
+
end
|
41
|
+
|
42
|
+
def basic_forward # :nodoc:
|
43
|
+
u = @waiting.pop
|
44
|
+
graph.each_adjacent(u) do |v|
|
45
|
+
@inDegrees[v] -= 1
|
46
|
+
@waiting.push(v) if @inDegrees[v].zero?
|
47
|
+
end
|
48
|
+
u
|
49
|
+
end
|
50
|
+
|
51
|
+
def at_beginning?; true; end # :nodoc: FIXME
|
52
|
+
def at_end?; @waiting.empty?; end # :nodoc:
|
53
|
+
|
54
|
+
end # class TopsortIterator
|
48
55
|
|
49
56
|
module Graph
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
57
|
+
|
58
|
+
# Returns a TopsortIterator.
|
59
|
+
|
60
|
+
def topsort_iterator
|
61
|
+
TopsortIterator.new(self)
|
62
|
+
end
|
63
|
+
|
64
|
+
# Returns true if the graph contains no cycles. This is only meaningful
|
65
|
+
# for directed graphs. Returns false for undirected graphs.
|
66
|
+
|
67
|
+
def acyclic?
|
68
|
+
topsort_iterator.length == num_vertices
|
69
|
+
end
|
70
|
+
|
71
|
+
end # module Graph
|
72
|
+
end # module RGL
|
@@ -1,34 +1,46 @@
|
|
1
|
+
# transitiv_closure.rb
|
2
|
+
#
|
1
3
|
# == transitive_closure
|
2
4
|
#
|
3
|
-
# The transitive closure of a graph G = (V,E) is a graph G* = (V,E*)
|
4
|
-
# contains an edge (u,v) if and only if G contains a path
|
5
|
-
# from u to v.
|
6
|
-
# the transitive closure graph tc.
|
5
|
+
# The transitive closure of a graph G = (V,E) is a graph G* = (V,E*),
|
6
|
+
# such that E* contains an edge (u,v) if and only if G contains a path
|
7
|
+
# (of at least one edge) from u to v. The transitive_closure() function
|
8
|
+
# transforms the input graph g into the transitive closure graph tc.
|
9
|
+
|
7
10
|
require 'rgl/adjacency'
|
8
11
|
|
9
12
|
module RGL
|
13
|
+
|
10
14
|
module Graph
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
end
|
15
|
+
|
16
|
+
# Floyd-Warshal algorithm which should be O(n^3), where n is the number of
|
17
|
+
# nodes. We can probably work a bit on the constant factors!
|
18
|
+
#
|
19
|
+
# In BGL, there is an algorithm with time complexity (worst-case) O(|V||E|)
|
20
|
+
# (see BOOST_DOC/transitive_closure.html), based on the detection of strong
|
21
|
+
# components.
|
22
|
+
|
23
|
+
def transitive_closure_floyd_warshal
|
24
|
+
raise NotDirectedError,
|
25
|
+
"transitive_closure makes sense only for directed graphs." unless directed?
|
26
|
+
tc = to_adjacency # direct links
|
27
|
+
|
28
|
+
# indirect links
|
29
|
+
|
30
|
+
each_vertex do |vi|
|
31
|
+
each_vertex do |vj|
|
32
|
+
each_vertex do |vk|
|
33
|
+
unless tc.has_edge?(vi, vj)
|
34
|
+
tc.add_edge(vi, vj) if has_edge?(vi, vk) and
|
35
|
+
has_edge?(vk, vj)
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
40
|
+
tc
|
41
|
+
end
|
42
|
+
|
43
|
+
alias_method :transitive_closure, :transitive_closure_floyd_warshal
|
44
|
+
|
45
|
+
end # module Graph
|
46
|
+
end # module RGL
|
data/lib/rgl/traversal.rb
CHANGED
@@ -1,296 +1,349 @@
|
|
1
|
-
#
|
2
|
-
# search. They are implemented as an RGL::GraphIterator which is a Stream of
|
3
|
-
# vertices of a given graph. The streams are not reversable.
|
1
|
+
# traversal.rb
|
4
2
|
#
|
5
|
-
#
|
6
|
-
# RGL::
|
7
|
-
#
|
8
|
-
#
|
3
|
+
# This file defines the basic graph traversal algorithm for DFS and BFS search.
|
4
|
+
# They are implemented as an RGL::GraphIterator, which is a Stream of vertices
|
5
|
+
# of a given graph. The streams are not reversable.
|
6
|
+
#
|
7
|
+
# Beside being an iterator in the sense of the Stream mixin, RGL::BFSIterator
|
8
|
+
# and RGL::DFSIterator follow the BGL
|
9
|
+
# Visitor[http://www.boost.org/libs/graph/doc/visitor_concepts.html] Concepts
|
10
|
+
# in a slightly modified fashion (especially for the RGL::DFSIterator).
|
9
11
|
|
10
12
|
require 'rgl/base'
|
11
13
|
require 'rubygems' rescue LoadError # If using stream gem
|
12
14
|
require 'stream'
|
13
15
|
|
14
16
|
module RGL
|
15
|
-
module GraphWrapper # :nodoc:
|
16
|
-
attr_accessor :graph
|
17
17
|
|
18
|
-
|
19
|
-
|
20
|
-
|
18
|
+
module GraphWrapper # :nodoc:
|
19
|
+
|
20
|
+
attr_accessor :graph
|
21
|
+
|
22
|
+
# Creates a new GraphWrapper on _graph_.
|
23
|
+
def initialize (graph)
|
24
|
+
@graph = graph
|
25
|
+
end
|
26
|
+
|
27
|
+
end # module GraphWrapper
|
21
28
|
|
22
|
-
# A GraphIterator is the abstract superclass of all Iterators on graphs.
|
23
|
-
# such iterator should implement the protocol defined in module Stream.
|
29
|
+
# A GraphIterator is the abstract superclass of all Iterators on graphs.
|
30
|
+
# Each such iterator should implement the protocol defined in module Stream.
|
24
31
|
module GraphIterator
|
25
|
-
|
26
|
-
|
32
|
+
include Stream
|
33
|
+
include GraphWrapper
|
27
34
|
end
|
28
35
|
|
29
|
-
# Module GraphVisitor defines
|
30
|
-
#
|
36
|
+
# Module GraphVisitor defines the BGL
|
37
|
+
# BFS[http://www.boost.org/libs/graph/doc/BFSVisitor.html] Visitor Concept).
|
31
38
|
#
|
32
|
-
# Visitors provide a mechanism for extending an algorithm
|
33
|
-
# what is done at each step of the algorithm.
|
34
|
-
# their own operations at various steps within a graph algorithm.
|
39
|
+
# Visitors provide a mechanism for extending an algorithm (i.e., for
|
40
|
+
# customizing what is done at each step of the algorithm). They allow users
|
41
|
+
# to insert their own operations at various steps within a graph algorithm.
|
35
42
|
#
|
36
43
|
# Graph algorithms typically have multiple event points where one may want to
|
37
|
-
# insert a call-back.
|
38
|
-
# to the various event points.
|
39
|
-
# points.
|
44
|
+
# insert a call-back. Therefore, visitors have several methods that
|
45
|
+
# correspond to the various event points. Each algorithm has a different
|
46
|
+
# set of event points. The following are common to both DFS and BFS search.
|
40
47
|
#
|
41
|
-
#
|
42
|
-
#
|
43
|
-
#
|
44
|
-
#
|
45
|
-
#
|
46
|
-
#
|
48
|
+
# * examine_vertex
|
49
|
+
# * finish_vertex
|
50
|
+
# * examine_edge
|
51
|
+
# * tree_edge
|
52
|
+
# * back_edge
|
53
|
+
# * forward_edge
|
47
54
|
#
|
48
|
-
# These methods are all called handle_* and can be set to appropriate blocks
|
55
|
+
# These methods are all called handle_* and can be set to appropriate blocks,
|
49
56
|
# using the methods set_*_event_handler, which are defined for each event
|
50
57
|
# mentioned above.
|
51
58
|
#
|
52
|
-
# As an alternative you can also
|
53
|
-
#
|
54
|
-
# TarjanSccVisitor).
|
59
|
+
# As an alternative, you can also override the handle_* methods in a
|
60
|
+
# subclass, to configure the algorithm (as an example, see TarjanSccVisitor).
|
55
61
|
#
|
56
|
-
# During a graph traversal vertices are *colored* using the colors :GRAY
|
57
|
-
# (when waiting) and :BLACK when finished.
|
62
|
+
# During a graph traversal, vertices are *colored* using the colors :GRAY
|
63
|
+
# (when waiting) and :BLACK when finished. All other vertices are :WHITE.
|
58
64
|
# The color_map is also maintained in the visitor.
|
65
|
+
|
59
66
|
module GraphVisitor
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
67
|
+
|
68
|
+
include GraphWrapper
|
69
|
+
|
70
|
+
attr_reader :color_map
|
71
|
+
|
72
|
+
# Create a new GraphVisitor on _graph_.
|
73
|
+
|
74
|
+
def initialize (graph)
|
75
|
+
super graph
|
76
|
+
reset
|
77
|
+
end
|
78
|
+
|
79
|
+
# Mark each vertex unvisited (i.e. :WHITE)
|
80
|
+
|
81
|
+
def reset
|
82
|
+
@color_map = Hash.new(:WHITE)
|
83
|
+
end
|
84
|
+
|
85
|
+
# Returns true if vertex _v_ is colored :BLACK (i.e. finished).
|
86
|
+
|
87
|
+
def finished_vertex? (v)
|
88
|
+
@color_map[v] == :BLACK
|
89
|
+
end
|
78
90
|
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
#
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
123
|
-
|
91
|
+
# Attach a map to the visitor which records the distance of a visited
|
92
|
+
# vertex to the start vertex.
|
93
|
+
#
|
94
|
+
# This is similar to BGLs
|
95
|
+
# distance_recorder[http://www.boost.org/libs/graph/doc/distance_recorder.html].
|
96
|
+
#
|
97
|
+
# After the distance_map is attached, the visitor has a new method
|
98
|
+
# distance_to_root, which answers the distance to the start vertex.
|
99
|
+
|
100
|
+
def attach_distance_map (map = Hash.new(0))
|
101
|
+
@dist_map = map
|
102
|
+
|
103
|
+
class << self
|
104
|
+
|
105
|
+
def handle_tree_edge (u, v)
|
106
|
+
super
|
107
|
+
@dist_map[v] = @dist_map[u] + 1
|
108
|
+
end
|
109
|
+
|
110
|
+
# Answer the distance to the start vertex.
|
111
|
+
|
112
|
+
def distance_to_root (v)
|
113
|
+
@dist_map[v]
|
114
|
+
end
|
115
|
+
|
116
|
+
end # class
|
117
|
+
end
|
118
|
+
|
119
|
+
# Shall we follow the edge (u,v); i.e. v has color :WHITE
|
120
|
+
|
121
|
+
def follow_edge? (u, v) # :nodoc:
|
122
|
+
@color_map[v] == :WHITE
|
123
|
+
end
|
124
|
+
|
125
|
+
# == Visitor Event Points
|
126
|
+
|
127
|
+
def self.def_event_handler (m)
|
128
|
+
params = m =~ /edge/ ? "u,v" : "u"
|
129
|
+
self.class_eval %{
|
130
|
+
def handle_#{m} (#{params})
|
131
|
+
@#{m}_event_handler.call(#{params}) if defined? @#{m}_event_handler
|
132
|
+
end
|
133
|
+
|
134
|
+
def set_#{m}_event_handler (&b)
|
135
|
+
@#{m}_event_handler = b
|
136
|
+
end
|
137
|
+
}
|
138
|
+
end
|
139
|
+
|
140
|
+
%w[examine_vertex finish_vertex examine_edge tree_edge back_edge
|
141
|
+
forward_edge].each do |m|
|
142
|
+
def_event_handler(m)
|
143
|
+
end
|
144
|
+
|
145
|
+
end # module GraphVisitor
|
124
146
|
|
125
147
|
# A BFSIterator can be used to traverse a graph from a given start vertex in
|
126
|
-
# breath first search order.
|
148
|
+
# breath first search order. Since the Iterator also mixins the GraphVisitor,
|
127
149
|
# it provides all event points defined there.
|
128
150
|
#
|
129
|
-
# The vertices which are not yet visited are
|
130
|
-
#
|
131
|
-
# (when waiting) and :BLACK when finished.
|
151
|
+
# The vertices which are not yet visited are held in the queue @waiting.
|
152
|
+
# During the traversal, vertices are *colored* using the colors :GRAY
|
153
|
+
# (when waiting) and :BLACK when finished. All other vertices are :WHITE.
|
132
154
|
#
|
133
|
-
# For more doc see the BGL
|
134
|
-
#
|
155
|
+
# For more doc see the BGL
|
156
|
+
# BFS[http://www.boost.org/libs/graph/doc/BFSVisitor.html] Visitor Concept .
|
135
157
|
#
|
136
158
|
# See the implementation of bfs_search_tree_from for an example usage.
|
159
|
+
|
137
160
|
class BFSIterator
|
138
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
159
|
-
|
160
|
-
|
161
|
-
|
162
|
-
|
163
|
-
|
164
|
-
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
178
|
-
|
179
|
-
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
|
186
|
-
|
187
|
-
|
188
|
-
|
189
|
-
|
190
|
-
|
161
|
+
|
162
|
+
include GraphIterator
|
163
|
+
include GraphVisitor
|
164
|
+
|
165
|
+
attr_accessor :start_vertex
|
166
|
+
|
167
|
+
# Create a new BFSIterator on _graph_, starting at vertex _start_.
|
168
|
+
|
169
|
+
def initialize (graph, start=graph.detect{ |x| true })
|
170
|
+
super(graph)
|
171
|
+
@start_vertex = start
|
172
|
+
set_to_begin
|
173
|
+
end
|
174
|
+
|
175
|
+
# Returns true if the @color_map has only one entry (for the start vertex).
|
176
|
+
|
177
|
+
def at_beginning? # :nodoc:
|
178
|
+
@color_map.size == 1
|
179
|
+
end
|
180
|
+
|
181
|
+
# Returns true if @waiting is empty.
|
182
|
+
|
183
|
+
def at_end?
|
184
|
+
@waiting.empty?
|
185
|
+
end
|
186
|
+
|
187
|
+
# Reset the iterator to the initial state (i.e. at_beginning? == true).
|
188
|
+
|
189
|
+
def set_to_begin
|
190
|
+
color_map[@start_vertex] = :GRAY
|
191
|
+
@waiting = [@start_vertex] # a queue
|
192
|
+
handle_tree_edge(nil, @start_vertex) # discovers start vertex
|
193
|
+
end
|
194
|
+
|
195
|
+
def basic_forward # :nodoc:
|
196
|
+
u = next_vertex
|
197
|
+
handle_examine_vertex(u)
|
198
|
+
graph.each_adjacent(u) { |v|
|
199
|
+
handle_examine_edge(u, v)
|
200
|
+
if follow_edge?(u, v) # (u,v) is a tree edge
|
201
|
+
handle_tree_edge(u, v) # also discovers v
|
202
|
+
color_map[v] = :GRAY # color of v was :WHITE
|
203
|
+
@waiting.push(v)
|
204
|
+
else # (u,v) is a non tree edge
|
205
|
+
if color_map[v] == :GRAY
|
206
|
+
handle_back_edge(u, v) # (u,v) has gray target
|
207
|
+
else
|
208
|
+
handle_forward_edge(u, v) # (u,v) has black target
|
209
|
+
end
|
210
|
+
end
|
211
|
+
}
|
212
|
+
color_map[u] = :BLACK
|
213
|
+
handle_finish_vertex(u) # finish vertex
|
214
|
+
u
|
215
|
+
end
|
216
|
+
|
217
|
+
protected
|
218
|
+
|
219
|
+
def next_vertex # :nodoc:
|
220
|
+
# waiting is a queue
|
221
|
+
@waiting.shift
|
222
|
+
end
|
223
|
+
end # class BFSIterator
|
224
|
+
|
191
225
|
|
192
226
|
module Graph
|
193
|
-
# Returns a BFSIterator staring at vertex _v_.
|
194
|
-
def bfs_iterator (v=self.detect {|x| true})
|
195
|
-
BFSIterator.new(self,v)
|
196
|
-
end
|
197
|
-
|
198
|
-
# Returns a DirectedAdjacencyGraph which represents a BFS search tree
|
199
|
-
# starting at _v_. This methods uses the tree_edge_event of BFSIterator to
|
200
|
-
# record all tree edges of the search tree in the result.
|
201
|
-
def bfs_search_tree_from(v)
|
202
|
-
require 'rgl/adjacency'
|
203
|
-
bfs = bfs_iterator(v)
|
204
|
-
tree = DirectedAdjacencyGraph.new
|
205
|
-
bfs.set_tree_edge_event_handler {|from, to|
|
206
|
-
tree.add_edge from, to
|
207
|
-
}
|
208
|
-
bfs.set_to_end # does the search
|
209
|
-
tree
|
210
|
-
end
|
211
|
-
end
|
212
227
|
|
213
|
-
|
214
|
-
|
228
|
+
# Returns a BFSIterator, starting at vertex _v_.
|
229
|
+
|
230
|
+
def bfs_iterator (v = self.detect { |x| true})
|
231
|
+
BFSIterator.new(self, v)
|
232
|
+
end
|
233
|
+
|
234
|
+
# Returns a DirectedAdjacencyGraph, which represents a BFS search tree
|
235
|
+
# starting at _v_. This method uses the tree_edge_event of BFSIterator
|
236
|
+
# to record all tree edges of the search tree in the result.
|
237
|
+
|
238
|
+
def bfs_search_tree_from (v)
|
239
|
+
require 'rgl/adjacency'
|
240
|
+
bfs = bfs_iterator(v)
|
241
|
+
tree = DirectedAdjacencyGraph.new
|
242
|
+
bfs.set_tree_edge_event_handler { |from, to|
|
243
|
+
tree.add_edge(from, to)
|
244
|
+
}
|
245
|
+
bfs.set_to_end # does the search
|
246
|
+
tree
|
247
|
+
end
|
248
|
+
|
249
|
+
end # module Graph
|
250
|
+
|
251
|
+
|
252
|
+
# Iterator for a depth first search, starting at a given vertex. The only
|
253
|
+
# difference from BFSIterator is that @waiting is a stack, instead of a queue.
|
215
254
|
#
|
216
|
-
# Note that this is different
|
255
|
+
# Note that this is different from DFSVisitor, which is used in the recursive
|
217
256
|
# version for depth first search (see depth_first_search).
|
257
|
+
|
218
258
|
class DFSIterator < BFSIterator
|
219
|
-
def next_vertex
|
220
|
-
# waiting is a stack
|
221
|
-
@waiting.pop
|
222
|
-
end
|
223
|
-
end
|
224
259
|
|
225
|
-
|
226
|
-
|
227
|
-
|
228
|
-
|
229
|
-
|
260
|
+
def next_vertex
|
261
|
+
# waiting is a stack
|
262
|
+
@waiting.pop
|
263
|
+
end
|
264
|
+
|
265
|
+
end # class DFSIterator
|
266
|
+
|
267
|
+
|
268
|
+
# A DFSVisitor is needed by the depth_first_search and depth_first_visit
|
269
|
+
# methods of a graph. Besides the eventpoint of GraphVisitor, it provides
|
270
|
+
# an additional eventpoint start_vertex, which is called when a
|
271
|
+
# depth_first_search starts a new subtree of the depth first forest that is
|
272
|
+
# defined by the search.
|
230
273
|
#
|
231
|
-
# Note that the discover_vertex event defined in the BGL
|
232
|
-
#
|
233
|
-
#
|
234
|
-
# DFSIterator and BFSIterator.
|
274
|
+
# Note that the discover_vertex event defined in the BGL
|
275
|
+
# DFSVisitor[http://www.boost.org/libs/graph/doc/DFSVisitor.html] is not
|
276
|
+
# this is also defined in the common mixin GraphVisitor of DFSVisitor,
|
277
|
+
# DFSIterator, and BFSIterator.
|
278
|
+
|
235
279
|
class DFSVisitor
|
236
|
-
include GraphVisitor
|
237
280
|
|
238
|
-
|
239
|
-
|
281
|
+
include GraphVisitor
|
282
|
+
|
283
|
+
GraphVisitor.def_event_handler("start_vertex")
|
284
|
+
|
285
|
+
end # class DFSVisitor
|
286
|
+
|
240
287
|
|
241
288
|
module Graph
|
242
|
-
|
243
|
-
|
244
|
-
|
245
|
-
|
246
|
-
|
247
|
-
|
248
|
-
|
249
|
-
|
250
|
-
|
251
|
-
|
252
|
-
|
253
|
-
|
254
|
-
|
255
|
-
|
256
|
-
|
257
|
-
|
258
|
-
|
259
|
-
|
260
|
-
|
261
|
-
|
262
|
-
|
263
|
-
|
264
|
-
|
265
|
-
|
266
|
-
|
267
|
-
|
268
|
-
|
269
|
-
|
270
|
-
|
271
|
-
|
272
|
-
|
273
|
-
|
274
|
-
|
275
|
-
|
276
|
-
|
277
|
-
|
278
|
-
|
279
|
-
|
280
|
-
|
281
|
-
|
282
|
-
|
289
|
+
|
290
|
+
# Returns a DFSIterator staring at vertex _v_.
|
291
|
+
|
292
|
+
def dfs_iterator (v = self.detect { |x| true })
|
293
|
+
DFSIterator.new(self, v)
|
294
|
+
end
|
295
|
+
|
296
|
+
# Do a recursive DFS search on the whole graph. If a block is passed,
|
297
|
+
# it is called on each _finish_vertex_ event. See
|
298
|
+
# strongly_connected_components for an example usage.
|
299
|
+
|
300
|
+
def depth_first_search (vis = DFSVisitor.new(self), &b)
|
301
|
+
each_vertex do |u|
|
302
|
+
unless vis.finished_vertex?(u)
|
303
|
+
vis.handle_start_vertex(u)
|
304
|
+
depth_first_visit(u, vis, &b)
|
305
|
+
end
|
306
|
+
end
|
307
|
+
end
|
308
|
+
|
309
|
+
# Start a depth first search at vertex _u_. The block _b_ is called on
|
310
|
+
# each finish_vertex event.
|
311
|
+
|
312
|
+
def depth_first_visit (u, vis = DFSVisitor.new(self), &b)
|
313
|
+
vis.color_map[u] = :GRAY
|
314
|
+
vis.handle_examine_vertex(u)
|
315
|
+
each_adjacent(u) { |v|
|
316
|
+
vis.handle_examine_edge(u, v)
|
317
|
+
if vis.follow_edge?(u, v) # (u,v) is a tree edge
|
318
|
+
vis.handle_tree_edge(u, v) # also discovers v
|
319
|
+
vis.color_map[v] = :GRAY # color of v was :WHITE
|
320
|
+
depth_first_visit(v, vis, &b)
|
321
|
+
else # (u,v) is a non tree edge
|
322
|
+
if vis.color_map[v] == :GRAY
|
323
|
+
vis.handle_back_edge(u, v) # (u,v) has gray target
|
324
|
+
else
|
325
|
+
vis.handle_forward_edge(u, v) # (u,v) is a cross or forward edge
|
326
|
+
end
|
327
|
+
end
|
328
|
+
}
|
329
|
+
vis.color_map[u] = :BLACK
|
330
|
+
vis.handle_finish_vertex(u) # finish vertex
|
331
|
+
b.call(u)
|
332
|
+
end
|
333
|
+
|
334
|
+
end # module Graph
|
283
335
|
|
284
336
|
=begin
|
285
337
|
def acyclic?
|
286
|
-
|
287
|
-
|
288
|
-
|
289
|
-
|
290
|
-
|
291
|
-
|
292
|
-
|
293
|
-
|
338
|
+
has_cycle = false
|
339
|
+
dfs = DFSIterator.new(self)
|
340
|
+
dfs.set_back_edge_event {has_cycle = true}
|
341
|
+
dfs_each(dfs) do |x|
|
342
|
+
puts x,has_cycle,dfs.inspect
|
343
|
+
return false if has_cycle
|
344
|
+
end
|
345
|
+
true
|
294
346
|
end
|
295
347
|
=end
|
296
|
-
|
348
|
+
|
349
|
+
end # module RGL
|