plexus 0.5.4 → 0.5.5
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/Gemfile +3 -0
- data/LICENSE +37 -0
- data/README.md +208 -0
- data/Rakefile +25 -0
- data/lib/plexus.rb +90 -0
- data/lib/plexus/adjacency_graph.rb +225 -0
- data/lib/plexus/arc.rb +60 -0
- data/lib/plexus/arc_number.rb +50 -0
- data/lib/plexus/biconnected.rb +84 -0
- data/lib/plexus/chinese_postman.rb +91 -0
- data/lib/plexus/classes/graph_classes.rb +28 -0
- data/lib/plexus/common.rb +63 -0
- data/lib/plexus/comparability.rb +63 -0
- data/lib/plexus/directed_graph.rb +78 -0
- data/lib/plexus/directed_graph/algorithms.rb +95 -0
- data/lib/plexus/directed_graph/distance.rb +167 -0
- data/lib/plexus/dot.rb +94 -0
- data/lib/plexus/edge.rb +38 -0
- data/lib/plexus/ext.rb +79 -0
- data/lib/plexus/graph.rb +628 -0
- data/lib/plexus/graph_api.rb +35 -0
- data/lib/plexus/labels.rb +112 -0
- data/lib/plexus/maximum_flow.rb +77 -0
- data/lib/plexus/ruby_compatibility.rb +17 -0
- data/lib/plexus/search.rb +510 -0
- data/lib/plexus/strong_components.rb +93 -0
- data/lib/plexus/support/support.rb +9 -0
- data/lib/plexus/undirected_graph.rb +56 -0
- data/lib/plexus/undirected_graph/algorithms.rb +90 -0
- data/lib/plexus/version.rb +6 -0
- data/spec/biconnected_spec.rb +27 -0
- data/spec/chinese_postman_spec.rb +27 -0
- data/spec/community_spec.rb +44 -0
- data/spec/complement_spec.rb +27 -0
- data/spec/digraph_distance_spec.rb +121 -0
- data/spec/digraph_spec.rb +339 -0
- data/spec/dot_spec.rb +48 -0
- data/spec/edge_spec.rb +158 -0
- data/spec/inspection_spec.rb +38 -0
- data/spec/multi_edge_spec.rb +32 -0
- data/spec/neighborhood_spec.rb +36 -0
- data/spec/properties_spec.rb +146 -0
- data/spec/search_spec.rb +227 -0
- data/spec/spec.opts +4 -0
- data/spec/spec_helper.rb +59 -0
- data/spec/strong_components_spec.rb +61 -0
- data/spec/triangulated_spec.rb +125 -0
- data/spec/undirected_graph_spec.rb +220 -0
- data/vendor/priority-queue/CHANGELOG +33 -0
- data/vendor/priority-queue/Makefile +140 -0
- data/vendor/priority-queue/README +133 -0
- data/vendor/priority-queue/benchmark/dijkstra.rb +171 -0
- data/vendor/priority-queue/compare_comments.rb +49 -0
- data/vendor/priority-queue/doc/c-vs-rb.png +0 -0
- data/vendor/priority-queue/doc/compare_big.gp +14 -0
- data/vendor/priority-queue/doc/compare_big.png +0 -0
- data/vendor/priority-queue/doc/compare_small.gp +15 -0
- data/vendor/priority-queue/doc/compare_small.png +0 -0
- data/vendor/priority-queue/doc/results.csv +37 -0
- data/vendor/priority-queue/ext/priority_queue/CPriorityQueue/extconf.rb +2 -0
- data/vendor/priority-queue/ext/priority_queue/CPriorityQueue/priority_queue.c +947 -0
- data/vendor/priority-queue/lib/priority_queue.rb +14 -0
- data/vendor/priority-queue/lib/priority_queue/c_priority_queue.rb +1 -0
- data/vendor/priority-queue/lib/priority_queue/poor_priority_queue.rb +46 -0
- data/vendor/priority-queue/lib/priority_queue/ruby_priority_queue.rb +526 -0
- data/vendor/priority-queue/priority_queue.so +0 -0
- data/vendor/priority-queue/setup.rb +1551 -0
- data/vendor/priority-queue/test/priority_queue_test.rb +371 -0
- data/vendor/rdot.rb +360 -0
- metadata +100 -10
data/lib/plexus/graph.rb
ADDED
@@ -0,0 +1,628 @@
|
|
1
|
+
module Plexus
|
2
|
+
# Using the methods required by the {GraphAPI}, it implements all the
|
3
|
+
# *basic* functions of a {Graph} using *only* functions
|
4
|
+
# requested in {GraphAPI}. The process is under the control of the pattern
|
5
|
+
# {AdjacencyGraphBuilder}, unless a specific implementation is specified
|
6
|
+
# during initialization.
|
7
|
+
#
|
8
|
+
# An actual, complete implementation still needs to be done using this cheap result,
|
9
|
+
# hence {Digraph}, {UndirectedGraph} and their roomates.
|
10
|
+
module GraphBuilder
|
11
|
+
include Enumerable
|
12
|
+
include Labels
|
13
|
+
include Dot
|
14
|
+
|
15
|
+
#def self.[](*a)
|
16
|
+
#puts self
|
17
|
+
#self.new.from_array(*a)
|
18
|
+
#end
|
19
|
+
# after the class->module transition, has been moved at implementation level,
|
20
|
+
# using a helper (extends_host)
|
21
|
+
#
|
22
|
+
extends_host
|
23
|
+
module ClassMethods
|
24
|
+
def [](*a)
|
25
|
+
self.new.from_array(*a)
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
# Creates a generic graph.
|
30
|
+
#
|
31
|
+
# @param [Hash(Plexus::Graph, Array)] *params initialization parameters.
|
32
|
+
# See {AdjacencyGraphBuilder#implementation_initialize} for more details.
|
33
|
+
# @return [Graph]
|
34
|
+
#
|
35
|
+
def initialize(*params)
|
36
|
+
raise ArgumentError if params.any? do |p|
|
37
|
+
# FIXME: checking wether it's a GraphBuilder (module) is not sufficient
|
38
|
+
# and the is_a? redefinition trick (instance_evaling) should be
|
39
|
+
# completed by a clever way to check the actual class of p.
|
40
|
+
# Maybe using ObjectSpace to get the available Graph classes?
|
41
|
+
!(p.is_a? Plexus::GraphBuilder or p.is_a? Array or p.is_a? Hash)
|
42
|
+
end
|
43
|
+
|
44
|
+
args = params.last || {}
|
45
|
+
|
46
|
+
class << self
|
47
|
+
self
|
48
|
+
end.module_eval do
|
49
|
+
# These inclusions trigger some validations checks by the way.
|
50
|
+
include(args[:implementation] ? args[:implementation] : Plexus::AdjacencyGraphBuilder)
|
51
|
+
include(args[:algorithmic_category] ? args[:algorithmic_category] : Plexus::DigraphBuilder )
|
52
|
+
include Plexus::GraphAPI
|
53
|
+
end
|
54
|
+
|
55
|
+
implementation_initialize(*params)
|
56
|
+
end
|
57
|
+
|
58
|
+
# Shortcut for creating a Graph.
|
59
|
+
#
|
60
|
+
# Using an arry of implicit {Arc}, specifying the vertices:
|
61
|
+
#
|
62
|
+
# Plexus::Graph[1,2, 2,3, 2,4, 4,5].edges.to_a.to_s
|
63
|
+
# # => "(1-2)(2-3)(2-4)(4-5)"
|
64
|
+
#
|
65
|
+
# Using a Hash for specifying labels along the way:
|
66
|
+
#
|
67
|
+
# Plexus::Graph[ [:a,:b] => 3, [:b,:c] => 4 ] (note: do not use for Multi or Pseudo graphs)
|
68
|
+
#
|
69
|
+
# @param [Array, Hash] *a
|
70
|
+
# @return [Graph]
|
71
|
+
def from_array(*a)
|
72
|
+
if a.size == 1 and a[0].is_a? Hash
|
73
|
+
# Convert to edge class
|
74
|
+
a[0].each do |k,v|
|
75
|
+
#FIXME, edge class shouldn't be assume here!!!
|
76
|
+
if edge_class.include? Plexus::ArcNumber
|
77
|
+
add_edge!(edge_class[k[0],k[1],nil,v])
|
78
|
+
else
|
79
|
+
add_edge!(edge_class[k[0],k[1],v])
|
80
|
+
end
|
81
|
+
end
|
82
|
+
#FIXME, edge class shouldn't be assume here!!!
|
83
|
+
elsif a[0].is_a? Plexus::Arc
|
84
|
+
a.each{ |e| add_edge!(e); self[e] = e.label}
|
85
|
+
elsif a.size % 2 == 0
|
86
|
+
0.step(a.size-1, 2) {|i| add_edge!(a[i], a[i+1])}
|
87
|
+
else
|
88
|
+
raise ArgumentError
|
89
|
+
end
|
90
|
+
self
|
91
|
+
end
|
92
|
+
|
93
|
+
# Non destructive version of {AdjacencyGraphBuilder#add_vertex!} (works on a copy of the graph).
|
94
|
+
#
|
95
|
+
# @param [vertex] v
|
96
|
+
# @param [Label] l
|
97
|
+
# @return [Graph] a new graph with the supplementary vertex
|
98
|
+
def add_vertex(v, l = nil)
|
99
|
+
x = self.class.new(self)
|
100
|
+
x.add_vertex!(v, l)
|
101
|
+
end
|
102
|
+
|
103
|
+
# Non destructive version {AdjacencyGraphBuilder#add_edge!} (works on a copy of the graph).
|
104
|
+
#
|
105
|
+
# @param [vertex] u
|
106
|
+
# @param [vertex] v
|
107
|
+
# @param [Label] l
|
108
|
+
# @return [Graph] a new graph with the supplementary edge
|
109
|
+
def add_edge(u, v = nil, l = nil)
|
110
|
+
x = self.class.new(self)
|
111
|
+
x.add_edge!(u, v, l)
|
112
|
+
end
|
113
|
+
alias add_arc add_edge
|
114
|
+
|
115
|
+
# Non destructive version of {AdjacencyGraphBuilder#remove_vertex!} (works on a copy of the graph).
|
116
|
+
#
|
117
|
+
# @param [vertex] v
|
118
|
+
# @return [Graph] a new graph without the specified vertex
|
119
|
+
def remove_vertex(v)
|
120
|
+
x = self.class.new(self)
|
121
|
+
x.remove_vertex!(v)
|
122
|
+
end
|
123
|
+
|
124
|
+
# Non destructive version {AdjacencyGraphBuilder#remove_edge!} (works on a copy of the graph).
|
125
|
+
#
|
126
|
+
# @param [vertex] u
|
127
|
+
# @param [vertex] v
|
128
|
+
# @return [Graph] a new graph without the specified edge
|
129
|
+
def remove_edge(u, v = nil)
|
130
|
+
x = self.class.new(self)
|
131
|
+
x.remove_edge!(u, v)
|
132
|
+
end
|
133
|
+
alias remove_arc remove_edge
|
134
|
+
|
135
|
+
# Computes the adjacent portions of the Graph.
|
136
|
+
#
|
137
|
+
# The options specify the parameters about the adjacency search.
|
138
|
+
# Note: it is probably more efficently done in the implementation class.
|
139
|
+
#
|
140
|
+
# @param [vertex, Edge] x can either be a vertex an edge
|
141
|
+
# @option options [Symbol] :type (:vertices) can be either `:edges` or `:vertices`
|
142
|
+
# @option options [Symbol] :direction (:all) can be `:in`, `:out` or `:all`
|
143
|
+
# @return [Array] an array of the adjacent portions
|
144
|
+
# @fixme
|
145
|
+
def adjacent(x, options = {})
|
146
|
+
d = directed? ? (options[:direction] || :out) : :all
|
147
|
+
|
148
|
+
# Discharge the easy ones first.
|
149
|
+
return [x.source] if x.is_a? Arc and options[:type] == :vertices and d == :in
|
150
|
+
return [x.target] if x.is_a? Arc and options[:type] == :vertices and d == :out
|
151
|
+
return [x.source, x.target] if x.is_a? Arc and options[:type] != :edges and d == :all
|
152
|
+
|
153
|
+
(options[:type] == :edges ? edges : to_a).select { |u| adjacent?(x,u,d) }
|
154
|
+
end
|
155
|
+
#FIXME: This is a hack around a serious problem
|
156
|
+
alias graph_adjacent adjacent
|
157
|
+
|
158
|
+
# Adds all specified vertices to the vertex set.
|
159
|
+
#
|
160
|
+
# @param [#each] *a an Enumerable vertices set
|
161
|
+
# @return [Graph] `self`
|
162
|
+
def add_vertices!(*a)
|
163
|
+
a.each { |v| add_vertex! v }
|
164
|
+
self
|
165
|
+
end
|
166
|
+
|
167
|
+
# Same as {GraphBuilder#add_vertices! add_vertices!} but works on copy of the receiver.
|
168
|
+
#
|
169
|
+
# @param [#each] *a
|
170
|
+
# @return [Graph] a modified copy of `self`
|
171
|
+
def add_vertices(*a)
|
172
|
+
x = self.class.new(self)
|
173
|
+
x.add_vertices!(*a)
|
174
|
+
self
|
175
|
+
end
|
176
|
+
|
177
|
+
# Adds all edges mentionned in the specified Enumerable to the edge set.
|
178
|
+
#
|
179
|
+
# Elements of the Enumerable can be either two-element arrays or instances of
|
180
|
+
# {Edge} or {Arc}.
|
181
|
+
#
|
182
|
+
# @param [#each] *a an Enumerable edges set
|
183
|
+
# @return [Graph] `self`
|
184
|
+
def add_edges!(*a)
|
185
|
+
a.each { |edge| add_edge!(edge) }
|
186
|
+
self
|
187
|
+
end
|
188
|
+
alias add_arcs! add_edges!
|
189
|
+
|
190
|
+
# Same as {GraphBuilder#add_egdes! add_edges!} but works on a copy of the receiver.
|
191
|
+
#
|
192
|
+
# @param [#each] *a an Enumerable edges set
|
193
|
+
# @return [Graph] a modified copy of `self`
|
194
|
+
def add_edges(*a)
|
195
|
+
x = self.class.new(self)
|
196
|
+
x.add_edges!(*a)
|
197
|
+
self
|
198
|
+
end
|
199
|
+
alias add_arcs add_edges
|
200
|
+
|
201
|
+
# Removes all vertices mentionned in the specified Enumerable from the graph.
|
202
|
+
#
|
203
|
+
# The process relies on {GraphBuilder#remove_vertex! remove_vertex!}.
|
204
|
+
#
|
205
|
+
# @param [#each] *a an Enumerable vertices set
|
206
|
+
# @return [Graph] `self`
|
207
|
+
def remove_vertices!(*a)
|
208
|
+
a.each { |v| remove_vertex! v }
|
209
|
+
end
|
210
|
+
alias delete_vertices! remove_vertices!
|
211
|
+
|
212
|
+
# Same as {GraphBuilder#remove_vertices! remove_vertices!} but works on a copy of the receiver.
|
213
|
+
#
|
214
|
+
# @param [#each] *a a vertex Enumerable set
|
215
|
+
# @return [Graph] a modified copy of `self`
|
216
|
+
def remove_vertices(*a)
|
217
|
+
x = self.class.new(self)
|
218
|
+
x.remove_vertices(*a)
|
219
|
+
end
|
220
|
+
alias delete_vertices remove_vertices
|
221
|
+
|
222
|
+
# Removes all edges mentionned in the specified Enumerable from the graph.
|
223
|
+
#
|
224
|
+
# The process relies on {GraphBuilder#remove_edges! remove_edges!}.
|
225
|
+
#
|
226
|
+
# @param [#each] *a an Enumerable edges set
|
227
|
+
# @return [Graph] `self`
|
228
|
+
def remove_edges!(*a)
|
229
|
+
a.each { |e| remove_edge! e }
|
230
|
+
end
|
231
|
+
alias remove_arcs! remove_edges!
|
232
|
+
alias delete_edges! remove_edges!
|
233
|
+
alias delete_arcs! remove_edges!
|
234
|
+
|
235
|
+
# Same as {GraphBuilder#remove_edges! remove_edges!} but works on a copy of the receiver.
|
236
|
+
#
|
237
|
+
# @param [#each] *a an Enumerable edges set
|
238
|
+
# @return [Graph] a modified copy of `self`
|
239
|
+
def remove_edges(*a)
|
240
|
+
x = self.class.new(self)
|
241
|
+
x.remove_edges!(*a)
|
242
|
+
end
|
243
|
+
alias remove_arcs remove_edges
|
244
|
+
alias delete_edges remove_edges
|
245
|
+
alias delete_arcs remove_edges
|
246
|
+
|
247
|
+
# Executes the given block for each vertex. It allows for mixing Enumerable in.
|
248
|
+
def each(&block)
|
249
|
+
vertices.each(&block)
|
250
|
+
end
|
251
|
+
|
252
|
+
# Returns true if the specified vertex belongs to the graph.
|
253
|
+
#
|
254
|
+
# This is a default implementation that is of O(n) average complexity.
|
255
|
+
# If a subclass uses a hash to store vertices, then this can be
|
256
|
+
# made into an O(1) average complexity operation.
|
257
|
+
#
|
258
|
+
# @param [vertex] v
|
259
|
+
# @return [Boolean]
|
260
|
+
def vertex?(v)
|
261
|
+
vertices.include?(v)
|
262
|
+
end
|
263
|
+
alias has_vertex? vertex?
|
264
|
+
# TODO: (has_)vertices?
|
265
|
+
|
266
|
+
# Returns true if u or (u,v) is an {Edge edge} of the graph.
|
267
|
+
#
|
268
|
+
# @overload edge?(a)
|
269
|
+
# @param [Arc, Edge] a
|
270
|
+
# @overload edge?(u, v)
|
271
|
+
# @param [vertex] u
|
272
|
+
# @param [vertex] v
|
273
|
+
# @return [Boolean]
|
274
|
+
def edge?(*args)
|
275
|
+
edges.include?(edge_convert(*args))
|
276
|
+
end
|
277
|
+
alias arc? edge?
|
278
|
+
alias has_edge? edge?
|
279
|
+
alias has_arc? edge?
|
280
|
+
|
281
|
+
# Tests two objects to see if they are adjacent.
|
282
|
+
#
|
283
|
+
# Note that in this method, one is primarily concerned with finding
|
284
|
+
# all adjacent objects in a graph to a given object. The concern is primarily on seeing
|
285
|
+
# if two objects touch. For two vertexes, any edge between the two will usually do, but
|
286
|
+
# the direction can be specified if needed.
|
287
|
+
#
|
288
|
+
# @param [vertex] source
|
289
|
+
# @param [vertex] target
|
290
|
+
# @param [Symbol] direction (:all) constraint on the direction of adjacency; may be either `:in`, `:out` or `:all`
|
291
|
+
def adjacent?(source, target, direction = :all)
|
292
|
+
if source.is_a? Plexus::Arc
|
293
|
+
raise NoArcError unless edge? source
|
294
|
+
if target.is_a? Plexus::Arc
|
295
|
+
raise NoArcError unless edge? target
|
296
|
+
(direction != :out and source.source == target.target) or (direction != :in and source.target == target.source)
|
297
|
+
else
|
298
|
+
raise NoVertexError unless vertex? target
|
299
|
+
(direction != :out and source.source == target) or (direction != :in and source.target == target)
|
300
|
+
end
|
301
|
+
else
|
302
|
+
raise NoVertexError unless vertex? source
|
303
|
+
if target.is_a? Plexus::Arc
|
304
|
+
raise NoArcError unless edge? target
|
305
|
+
(direction != :out and source == target.target) or (direction != :in and source == target.source)
|
306
|
+
else
|
307
|
+
raise NoVertexError unless vertex? target
|
308
|
+
(direction != :out and edge?(target,source)) or (direction != :in and edge?(source,target))
|
309
|
+
end
|
310
|
+
end
|
311
|
+
end
|
312
|
+
|
313
|
+
# Is the graph connected?
|
314
|
+
#
|
315
|
+
# A graph is called connected if every pair of distinct vertices in the graph
|
316
|
+
# can be connected through some path. The exact definition depends on whether
|
317
|
+
# the graph is directed or not, hence this method should overriden in specific
|
318
|
+
# implementations.
|
319
|
+
#
|
320
|
+
# This methods implements a lazy routine using the internal vertices hash.
|
321
|
+
# If you ever want to check connectivity state using a bfs/dfs algorithm, use
|
322
|
+
# the `:algo => :bfs` or `:dfs` option.
|
323
|
+
#
|
324
|
+
# @return [Boolean] `true` if the graph is connected, `false` otherwise
|
325
|
+
def connected?(options = {})
|
326
|
+
options = options.reverse_merge! :algo => :bfs
|
327
|
+
if options[:algo] == (:bfs || :dfs)
|
328
|
+
num_nodes = 0
|
329
|
+
send(options[:algo]) { |n| num_nodes += 1 }
|
330
|
+
return num_nodes == @vertex_dict.size
|
331
|
+
else
|
332
|
+
!@vertex_dict.collect { |v| degree(v) > 0 }.any? { |check| check == false }
|
333
|
+
end
|
334
|
+
end
|
335
|
+
# TODO: do it!
|
336
|
+
# TODO: for directed graphs, add weakly_connected? and strongly_connected? (aliased as strong?)
|
337
|
+
# TODO: in the context of vertices/Arc, add connected_vertices? and disconnected_vertices?
|
338
|
+
# TODO: maybe implement some routine which would compute cuts and connectivity? tricky though,
|
339
|
+
# but would be useful (k_connected?(k))
|
340
|
+
|
341
|
+
# Returns true if the graph has no vertex.
|
342
|
+
#
|
343
|
+
# @return [Boolean]
|
344
|
+
def empty?
|
345
|
+
vertices.size.zero?
|
346
|
+
end
|
347
|
+
|
348
|
+
# Returns true if the given object is a vertex or an {Arc arc} of the graph.
|
349
|
+
#
|
350
|
+
# @param [vertex, Arc] x
|
351
|
+
def include?(x)
|
352
|
+
x.is_a?(Plexus::Arc) ? edge?(x) : vertex?(x)
|
353
|
+
end
|
354
|
+
alias has? include?
|
355
|
+
|
356
|
+
# Returns the neighborhood of the given vertex or {Arc arc}.
|
357
|
+
#
|
358
|
+
# This is equivalent to {GraphBuilder#adjacent adjacent}, but the type is based on the
|
359
|
+
# type of the specified object.
|
360
|
+
#
|
361
|
+
# @param [vertex, Arc] x
|
362
|
+
# @param [Symbol] direction (:all) can be either `:all`, `:in` or `:out`
|
363
|
+
def neighborhood(x, direction = :all)
|
364
|
+
adjacent(x, :direction => direction, :type => ((x.is_a? Plexus::Arc) ? :edges : :vertices ))
|
365
|
+
end
|
366
|
+
|
367
|
+
# Union of all neighborhoods of vertices (or edges) in the Enumerable x minus the contents of x.
|
368
|
+
#
|
369
|
+
# Definition taken from: Jorgen Bang-Jensen, Gregory Gutin, *Digraphs: Theory, Algorithms and Applications*, pg. 4
|
370
|
+
#
|
371
|
+
# @param [vertex] x
|
372
|
+
# @param [Symbol] direction can be either `:all`, `:in` or `:out`
|
373
|
+
def set_neighborhood(x, direction = :all)
|
374
|
+
x.inject(Set.new) { |a,v| a.merge(neighborhood(v, direction))}.reject { |v2| x.include?(v2) }
|
375
|
+
end
|
376
|
+
|
377
|
+
# Union of all {GraphBuilder#set_neighborhood set_neighborhoods} reachable
|
378
|
+
# among the specified edges.
|
379
|
+
#
|
380
|
+
# Definition taken from Jorgen Bang-Jensen, Gregory Gutin, *Digraphs:
|
381
|
+
# Theory, Algorithms and Applications*, pg. 46
|
382
|
+
#
|
383
|
+
# @param [vertex] w
|
384
|
+
# @param [Edges] p
|
385
|
+
# @param [Symbol] direction can be `:all`, `:in`, or `:out`
|
386
|
+
def closed_pth_neighborhood(w, p, direction = :all)
|
387
|
+
if p <= 0
|
388
|
+
w
|
389
|
+
elsif p == 1
|
390
|
+
(w + set_neighborhood(w, direction)).uniq
|
391
|
+
else
|
392
|
+
n = set_neighborhood(w, direction)
|
393
|
+
(w + n + closed_pth_neighborhood(n, p-1, direction)).uniq
|
394
|
+
end
|
395
|
+
end
|
396
|
+
|
397
|
+
# Returns the neighboorhoods reachable in a certain amount of steps from
|
398
|
+
# every vertex (or edge) in the specified Enumerable.
|
399
|
+
#
|
400
|
+
# Definition taken from Jorgen Bang-Jensen, Gregory Gutin, _Digraphs:
|
401
|
+
# Theory, Algorithms and Applications_, pg. 46
|
402
|
+
#
|
403
|
+
# @param [Enumerable] x
|
404
|
+
# @param [Integer] p number of steps to perform
|
405
|
+
# @param [Symbol] direction can be `:all`, `:in`, or `:out`
|
406
|
+
def open_pth_neighborhood(x, p, direction = :all)
|
407
|
+
if p <= 0
|
408
|
+
x
|
409
|
+
elsif p == 1
|
410
|
+
set_neighborhood(x,direction)
|
411
|
+
else
|
412
|
+
set_neighborhood(open_pth_neighborhood(x, p-1, direction), direction) -
|
413
|
+
closed_pth_neighborhood(x, p-1, direction)
|
414
|
+
end
|
415
|
+
end
|
416
|
+
|
417
|
+
# Returns the number of out-edges (for directed graphs) or the number of
|
418
|
+
# incident edges (for undirected graphs) of the specified vertex.
|
419
|
+
#
|
420
|
+
# @param [vertex] v
|
421
|
+
# @return [Integer] number of matching edges
|
422
|
+
def out_degree(v)
|
423
|
+
adjacent(v, :direction => :out).size
|
424
|
+
end
|
425
|
+
|
426
|
+
# Returns the number of in-edges (for directed graphs) or the number of
|
427
|
+
# incident edges (for undirected graphs) of the specified vertex
|
428
|
+
#
|
429
|
+
# @param [vertex] v
|
430
|
+
# @return [Integer] number of matching edges
|
431
|
+
def in_degree(v)
|
432
|
+
adjacent(v, :direction => :in).size
|
433
|
+
end
|
434
|
+
|
435
|
+
# Returns the sum of the number in and out edges for the specified vertex.
|
436
|
+
#
|
437
|
+
# @param [vertex] v
|
438
|
+
# @return [Integer] degree
|
439
|
+
def degree(v)
|
440
|
+
in_degree(v) + out_degree(v)
|
441
|
+
end
|
442
|
+
|
443
|
+
# Minimum in-degree of the graph.
|
444
|
+
#
|
445
|
+
# @return [Integer, nil] returns `nil` if the graph is empty
|
446
|
+
def min_in_degree
|
447
|
+
return nil if to_a.empty?
|
448
|
+
to_a.map { |v| in_degree(v) }.min
|
449
|
+
end
|
450
|
+
|
451
|
+
# Minimum out-degree of the graph.
|
452
|
+
#
|
453
|
+
# @return [Integer, nil] returns `nil` if the graph is empty
|
454
|
+
def min_out_degree
|
455
|
+
return nil if to_a.empty?
|
456
|
+
to_a.map {|v| out_degree(v)}.min
|
457
|
+
end
|
458
|
+
|
459
|
+
# Minimum degree of all vertexes of the graph.
|
460
|
+
#
|
461
|
+
# @return [Integer] `min` between {GraphBuilder#min_in_degree min_in_degree}
|
462
|
+
# and {GraphBuilder#min_out_degree max_out_degree}
|
463
|
+
def min_degree
|
464
|
+
[min_in_degree, min_out_degree].min
|
465
|
+
end
|
466
|
+
|
467
|
+
# Maximum in-degree of the graph.
|
468
|
+
#
|
469
|
+
# @return [Integer, nil] returns `nil` if the graph is empty
|
470
|
+
def max_in_degree
|
471
|
+
return nil if to_a.empty?
|
472
|
+
vertices.map { |v| in_degree(v)}.max
|
473
|
+
end
|
474
|
+
|
475
|
+
# Maximum out-degree of the graph.
|
476
|
+
#
|
477
|
+
# @return [Integer, nil] returns nil if the graph is empty
|
478
|
+
def max_out_degree
|
479
|
+
return nil if to_a.empty?
|
480
|
+
vertices.map { |v| out_degree(v)}.max
|
481
|
+
end
|
482
|
+
|
483
|
+
# Maximum degree of all vertexes of the graph.
|
484
|
+
#
|
485
|
+
# @return [Integer] `max` between {GraphBuilder#max_in_degree max_in_degree}
|
486
|
+
# and {GraphBuilder#max_out_degree max_out_degree}
|
487
|
+
def max_degree
|
488
|
+
[max_in_degree, max_out_degree].max
|
489
|
+
end
|
490
|
+
|
491
|
+
# Is the graph regular, that is are its min degree and max degree equal?
|
492
|
+
#
|
493
|
+
# @return [Boolean]
|
494
|
+
def regular?
|
495
|
+
min_degree == max_degree
|
496
|
+
end
|
497
|
+
|
498
|
+
# Number of vertices.
|
499
|
+
#
|
500
|
+
# @return [Integer]
|
501
|
+
def size
|
502
|
+
vertices.size
|
503
|
+
end
|
504
|
+
alias num_vertices size
|
505
|
+
alias number_of_vertices size
|
506
|
+
|
507
|
+
# Number of vertices.
|
508
|
+
#
|
509
|
+
# @return [Integer]
|
510
|
+
def num_vertices
|
511
|
+
vertices.size
|
512
|
+
end
|
513
|
+
alias number_of_vertices num_vertices
|
514
|
+
|
515
|
+
# Number of edges.
|
516
|
+
#
|
517
|
+
# @return [Integer]
|
518
|
+
def num_edges
|
519
|
+
edges.size
|
520
|
+
end
|
521
|
+
alias number_of_edges num_edges
|
522
|
+
|
523
|
+
# Utility method to show a string representation of the edges of the graph.
|
524
|
+
#def to_s
|
525
|
+
#edges.to_s
|
526
|
+
#end
|
527
|
+
|
528
|
+
# Equality is defined to be same set of edges and directed?
|
529
|
+
def eql?(g)
|
530
|
+
return false unless g.is_a? Plexus::Graph
|
531
|
+
|
532
|
+
(directed? == g.directed?) and
|
533
|
+
(vertices.sort == g.vertices.sort) and
|
534
|
+
(edges.sort == g.edges.sort)
|
535
|
+
end
|
536
|
+
alias == eql?
|
537
|
+
|
538
|
+
# Merges another graph into the receiver.
|
539
|
+
#
|
540
|
+
# @param [Graph] other the graph to merge in
|
541
|
+
# @return [Graph] `self`
|
542
|
+
def merge(other)
|
543
|
+
other.vertices.each { |v| add_vertex!(v) }
|
544
|
+
other.edges.each { |e| add_edge!(e) }
|
545
|
+
other.edges.each { |e| add_edge!(e.reverse) } if directed? and !other.directed?
|
546
|
+
self
|
547
|
+
end
|
548
|
+
|
549
|
+
# A synonym for {GraphBuilder#merge merge}, but doesn't modify the current graph.
|
550
|
+
#
|
551
|
+
# @param [Graph, Arc] other
|
552
|
+
# @return [Graph] a new graph
|
553
|
+
def +(other)
|
554
|
+
result = self.class.new(self)
|
555
|
+
case other
|
556
|
+
when Plexus::Graph
|
557
|
+
result.merge(other)
|
558
|
+
when Plexus::Arc
|
559
|
+
result.add_edge!(other)
|
560
|
+
else
|
561
|
+
result.add_vertex!(other)
|
562
|
+
end
|
563
|
+
end
|
564
|
+
|
565
|
+
# Removes all vertices in the specified graph.
|
566
|
+
#
|
567
|
+
# @param [Graph, Arc] other
|
568
|
+
# @return [Graph]
|
569
|
+
def -(other)
|
570
|
+
case other
|
571
|
+
when Plexus::Graph
|
572
|
+
induced_subgraph(vertices - other.vertices)
|
573
|
+
when Plexus::Arc
|
574
|
+
self.class.new(self).remove_edge!(other)
|
575
|
+
else
|
576
|
+
self.class.new(self).remove_vertex!(other)
|
577
|
+
end
|
578
|
+
end
|
579
|
+
|
580
|
+
# A synonym for {AdjacencyGraphBuilder#add_edge! add_edge!}.
|
581
|
+
def <<(edge)
|
582
|
+
add_edge!(edge)
|
583
|
+
end
|
584
|
+
|
585
|
+
# Computes the complement of the current graph.
|
586
|
+
#
|
587
|
+
# @return [Graph]
|
588
|
+
def complement
|
589
|
+
vertices.inject(self.class.new) do |a,v|
|
590
|
+
a.add_vertex!(v)
|
591
|
+
vertices.each { |v2| a.add_edge!(v, v2) unless edge?(v, v2) }; a
|
592
|
+
end
|
593
|
+
end
|
594
|
+
|
595
|
+
# Given an array of vertices, computes the induced subgraph.
|
596
|
+
#
|
597
|
+
# @param [Array(vertex)] v
|
598
|
+
# @return [Graph]
|
599
|
+
def induced_subgraph(v)
|
600
|
+
edges.inject(self.class.new) do |a,e|
|
601
|
+
(v.include?(e.source) and v.include?(e.target)) ? (a << e) : a
|
602
|
+
end
|
603
|
+
end
|
604
|
+
|
605
|
+
def inspect
|
606
|
+
## FIXME: broken, it's not updated. The issue's not with inspect, but it's worth mentionning here.
|
607
|
+
## Example:
|
608
|
+
## dg = Digraph[1,2, 2,3, 2,4, 4,5, 6,4, 1,6]
|
609
|
+
## dg.add_vertices! 1, 5, "yosh"
|
610
|
+
## # => Plexus::Digraph[Plexus::Arc[1,2,nil], Plexus::Arc[1,6,nil], Plexus::Arc[2,3,nil], Plexus::Arc[2,4,nil], Plexus::Arc[4,5,nil], Plexus::Arc[6,4,nil]]
|
611
|
+
## dg.vertex?("yosh")
|
612
|
+
## # => true
|
613
|
+
## dg
|
614
|
+
## # =>Plexus::Digraph[Plexus::Arc[1,2,nil], Plexus::Arc[1,6,nil], Plexus::Arc[2,3,nil], Plexus::Arc[2,4,nil], Plexus::Arc[4,5,nil], Plexus::Arc[6,4,nil]]
|
615
|
+
## the new vertex doesn't show up.
|
616
|
+
## Actually this version of inspect is far too verbose IMO :)
|
617
|
+
l = vertices.select { |v| self[v]}.map { |u| "vertex_label_set(#{u.inspect}, #{self[u].inspect})"}.join('.')
|
618
|
+
self.class.to_s + '[' + edges.map {|e| e.inspect}.join(', ') + ']' + (l && l != '' ? '.'+l : '')
|
619
|
+
end
|
620
|
+
|
621
|
+
private
|
622
|
+
|
623
|
+
# ?
|
624
|
+
def edge_convert(*args)
|
625
|
+
args[0].is_a?(Plexus::Arc) ? args[0] : edge_class[*args]
|
626
|
+
end
|
627
|
+
end
|
628
|
+
end
|