author | wenzelm |
Mon, 01 Mar 2021 22:22:12 +0100 | |
changeset 73340 | 0ffcad1f6130 |
parent 72065 | 11dc8929832d |
child 73359 | d8a0e996614b |
permissions | -rw-r--r-- |
46611 | 1 |
/* Title: Pure/General/graph.scala |
2 |
Author: Makarius |
|
3 |
||
4 |
Directed graphs. |
|
5 |
*/ |
|
6 |
||
7 |
package isabelle |
|
8 |
||
9 |
||
46661
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
10 |
import scala.collection.immutable.{SortedMap, SortedSet} |
46611 | 11 |
import scala.annotation.tailrec |
12 |
||
13 |
||
14 |
object Graph |
|
15 |
{ |
|
48348 | 16 |
class Duplicate[Key](val key: Key) extends Exception |
70651 | 17 |
{ |
18 |
override def toString: String = "Graph.Duplicate(" + key.toString + ")" |
|
19 |
} |
|
48348 | 20 |
class Undefined[Key](val key: Key) extends Exception |
70651 | 21 |
{ |
22 |
override def toString: String = "Graph.Undefined(" + key.toString + ")" |
|
23 |
} |
|
48348 | 24 |
class Cycles[Key](val cycles: List[List[Key]]) extends Exception |
70651 | 25 |
{ |
26 |
override def toString: String = cycles.mkString("Graph.Cycles(", ",\n", ")") |
|
27 |
} |
|
46611 | 28 |
|
46661
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
29 |
def empty[Key, A](implicit ord: Ordering[Key]): Graph[Key, A] = |
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
30 |
new Graph[Key, A](SortedMap.empty(ord)) |
46667 | 31 |
|
70636 | 32 |
def make[Key, A](entries: List[((Key, A), List[Key])], |
70692 | 33 |
symmetric: Boolean = false)(implicit ord: Ordering[Key]): Graph[Key, A] = |
49560 | 34 |
{ |
35 |
val graph1 = |
|
70692 | 36 |
(empty[Key, A](ord) /: entries) { case (graph, ((x, info), _)) => graph.new_node(x, info) } |
49560 | 37 |
val graph2 = |
67935 | 38 |
(graph1 /: entries) { case (graph, ((x, _), ys)) => |
39 |
(graph /: ys)({ case (g, y) => if (symmetric) g.add_edge(y, x) else g.add_edge(x, y) }) } |
|
49560 | 40 |
graph2 |
41 |
} |
|
42 |
||
46667 | 43 |
def string[A]: Graph[String, A] = empty(Ordering.String) |
44 |
def int[A]: Graph[Int, A] = empty(Ordering.Int) |
|
45 |
def long[A]: Graph[Long, A] = empty(Ordering.Long) |
|
49560 | 46 |
|
47 |
||
48 |
/* XML data representation */ |
|
49 |
||
50 |
def encode[Key, A](key: XML.Encode.T[Key], info: XML.Encode.T[A]): XML.Encode.T[Graph[Key, A]] = |
|
60215 | 51 |
(graph: Graph[Key, A]) => { |
49560 | 52 |
import XML.Encode._ |
53 |
list(pair(pair(key, info), list(key)))(graph.dest) |
|
60215 | 54 |
} |
49560 | 55 |
|
59245
be4180f3c236
more formal Graph_Display.Node (with ordering) and Graph_Display.Edge;
wenzelm
parents:
59212
diff
changeset
|
56 |
def decode[Key, A](key: XML.Decode.T[Key], info: XML.Decode.T[A])( |
59212 | 57 |
implicit ord: Ordering[Key]): XML.Decode.T[Graph[Key, A]] = |
60215 | 58 |
(body: XML.Body) => { |
49560 | 59 |
import XML.Decode._ |
59245
be4180f3c236
more formal Graph_Display.Node (with ordering) and Graph_Display.Edge;
wenzelm
parents:
59212
diff
changeset
|
60 |
make(list(pair(pair(key, info), list(key)))(body))(ord) |
60215 | 61 |
} |
46611 | 62 |
} |
63 |
||
64 |
||
46712 | 65 |
final class Graph[Key, A] private(rep: SortedMap[Key, (A, (SortedSet[Key], SortedSet[Key]))]) |
46611 | 66 |
{ |
46661
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
67 |
type Keys = SortedSet[Key] |
46611 | 68 |
type Entry = (A, (Keys, Keys)) |
69 |
||
46661
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
70 |
def ordering: Ordering[Key] = rep.ordering |
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
71 |
def empty_keys: Keys = SortedSet.empty[Key](ordering) |
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
72 |
|
46666
b01b6977a5e8
clarified signature -- avoid oddities of Iterable like Iterator.map;
wenzelm
parents:
46661
diff
changeset
|
73 |
|
b01b6977a5e8
clarified signature -- avoid oddities of Iterable like Iterator.map;
wenzelm
parents:
46661
diff
changeset
|
74 |
/* graphs */ |
46611 | 75 |
|
76 |
def is_empty: Boolean = rep.isEmpty |
|
48507 | 77 |
def defined(x: Key): Boolean = rep.isDefinedAt(x) |
68321 | 78 |
def domain: Set[Key] = rep.keySet |
46611 | 79 |
|
68496 | 80 |
def size: Int = rep.size |
56372
fadb0fef09d7
more explicit iterator terminology, in accordance to Scala 2.8 library;
wenzelm
parents:
50452
diff
changeset
|
81 |
def iterator: Iterator[(Key, Entry)] = rep.iterator |
fadb0fef09d7
more explicit iterator terminology, in accordance to Scala 2.8 library;
wenzelm
parents:
50452
diff
changeset
|
82 |
|
fadb0fef09d7
more explicit iterator terminology, in accordance to Scala 2.8 library;
wenzelm
parents:
50452
diff
changeset
|
83 |
def keys_iterator: Iterator[Key] = iterator.map(_._1) |
fadb0fef09d7
more explicit iterator terminology, in accordance to Scala 2.8 library;
wenzelm
parents:
50452
diff
changeset
|
84 |
def keys: List[Key] = keys_iterator.toList |
46611 | 85 |
|
49560 | 86 |
def dest: List[((Key, A), List[Key])] = |
56372
fadb0fef09d7
more explicit iterator terminology, in accordance to Scala 2.8 library;
wenzelm
parents:
50452
diff
changeset
|
87 |
(for ((x, (i, (_, succs))) <- iterator) yield ((x, i), succs.toList)).toList |
46611 | 88 |
|
46666
b01b6977a5e8
clarified signature -- avoid oddities of Iterable like Iterator.map;
wenzelm
parents:
46661
diff
changeset
|
89 |
override def toString: String = |
49560 | 90 |
dest.map({ case ((x, _), ys) => |
91 |
x.toString + " -> " + ys.iterator.map(_.toString).mkString("{", ", ", "}") }) |
|
46666
b01b6977a5e8
clarified signature -- avoid oddities of Iterable like Iterator.map;
wenzelm
parents:
46661
diff
changeset
|
92 |
.mkString("Graph(", ", ", ")") |
46611 | 93 |
|
94 |
private def get_entry(x: Key): Entry = |
|
95 |
rep.get(x) match { |
|
96 |
case Some(entry) => entry |
|
97 |
case None => throw new Graph.Undefined(x) |
|
98 |
} |
|
99 |
||
100 |
private def map_entry(x: Key, f: Entry => Entry): Graph[Key, A] = |
|
101 |
new Graph[Key, A](rep + (x -> f(get_entry(x)))) |
|
102 |
||
103 |
||
104 |
/* nodes */ |
|
105 |
||
106 |
def get_node(x: Key): A = get_entry(x)._1 |
|
107 |
||
108 |
def map_node(x: Key, f: A => A): Graph[Key, A] = |
|
109 |
map_entry(x, { case (i, ps) => (f(i), ps) }) |
|
110 |
||
111 |
||
112 |
/* reachability */ |
|
113 |
||
70764
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
114 |
/*reachable nodes with length of longest path*/ |
70794 | 115 |
def reachable_length( |
70800 | 116 |
count: Key => Long, |
70794 | 117 |
next: Key => Keys, |
70800 | 118 |
xs: List[Key]): Map[Key, Long] = |
70764
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
119 |
{ |
70800 | 120 |
def reach(length: Long)(rs: Map[Key, Long], x: Key): Map[Key, Long] = |
70764
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
121 |
if (rs.get(x) match { case Some(n) => n >= length case None => false }) rs |
70794 | 122 |
else ((rs + (x -> length)) /: next(x))(reach(length + count(x))) |
70800 | 123 |
(Map.empty[Key, Long] /: xs)(reach(0)) |
70764
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
124 |
} |
70800 | 125 |
def node_height(count: Key => Long): Map[Key, Long] = |
71601 | 126 |
reachable_length(count, imm_preds, maximals) |
70800 | 127 |
def node_depth(count: Key => Long): Map[Key, Long] = |
71601 | 128 |
reachable_length(count, imm_succs, minimals) |
70764
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
129 |
|
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
130 |
/*reachable nodes with size limit*/ |
70772
030a6baa5cb2
support headless_load_limit for more scalable load process;
wenzelm
parents:
70764
diff
changeset
|
131 |
def reachable_limit( |
70800 | 132 |
limit: Long, |
133 |
count: Key => Long, |
|
70772
030a6baa5cb2
support headless_load_limit for more scalable load process;
wenzelm
parents:
70764
diff
changeset
|
134 |
next: Key => Keys, |
030a6baa5cb2
support headless_load_limit for more scalable load process;
wenzelm
parents:
70764
diff
changeset
|
135 |
xs: List[Key]): Keys = |
70764
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
136 |
{ |
70800 | 137 |
def reach(res: (Long, Keys), x: Key): (Long, Keys) = |
70764
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
138 |
{ |
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
139 |
val (n, rs) = res |
70772
030a6baa5cb2
support headless_load_limit for more scalable load process;
wenzelm
parents:
70764
diff
changeset
|
140 |
if (rs.contains(x)) res |
71601 | 141 |
else ((n + count(x), rs + x) /: next(x))(reach) |
70764
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
142 |
} |
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
143 |
|
70800 | 144 |
@tailrec def reachs(size: Long, rs: Keys, rest: List[Key]): Keys = |
70764
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
145 |
rest match { |
70772
030a6baa5cb2
support headless_load_limit for more scalable load process;
wenzelm
parents:
70764
diff
changeset
|
146 |
case Nil => rs |
70764
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
147 |
case head :: tail => |
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
148 |
val (size1, rs1) = reach((size, rs), head) |
70772
030a6baa5cb2
support headless_load_limit for more scalable load process;
wenzelm
parents:
70764
diff
changeset
|
149 |
if (size > 0 && size1 > limit) rs |
70764
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
150 |
else reachs(size1, rs1, tail) |
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
151 |
} |
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
152 |
|
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
153 |
reachs(0, empty_keys, xs) |
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
154 |
} |
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
155 |
|
6d36b30fdd67
more operations -- incremental exploration of reachable nodes;
wenzelm
parents:
70699
diff
changeset
|
156 |
/*reachable nodes with result in topological order (for acyclic graphs)*/ |
72065
11dc8929832d
clarified order --- proper sorting of requirements;
wenzelm
parents:
71601
diff
changeset
|
157 |
private def reachable(next: Key => Keys, xs: List[Key], rev: Boolean = false): (List[List[Key]], Keys) = |
46611 | 158 |
{ |
48350
09bf3b73e446
clarified topological ordering: preserve order of adjacency via reverse fold;
wenzelm
parents:
48348
diff
changeset
|
159 |
def reach(x: Key, reached: (List[Key], Keys)): (List[Key], Keys) = |
46611 | 160 |
{ |
161 |
val (rs, r_set) = reached |
|
162 |
if (r_set(x)) reached |
|
163 |
else { |
|
72065
11dc8929832d
clarified order --- proper sorting of requirements;
wenzelm
parents:
71601
diff
changeset
|
164 |
val (rs1, r_set1) = |
11dc8929832d
clarified order --- proper sorting of requirements;
wenzelm
parents:
71601
diff
changeset
|
165 |
if (rev) ((rs, r_set + x) /: next(x))({ case (b, a) => reach(a, b) }) |
11dc8929832d
clarified order --- proper sorting of requirements;
wenzelm
parents:
71601
diff
changeset
|
166 |
else (next(x) :\ (rs, r_set + x))(reach) |
46611 | 167 |
(x :: rs1, r_set1) |
168 |
} |
|
169 |
} |
|
170 |
def reachs(reached: (List[List[Key]], Keys), x: Key): (List[List[Key]], Keys) = |
|
171 |
{ |
|
172 |
val (rss, r_set) = reached |
|
48350
09bf3b73e446
clarified topological ordering: preserve order of adjacency via reverse fold;
wenzelm
parents:
48348
diff
changeset
|
173 |
val (rs, r_set1) = reach(x, (Nil, r_set)) |
46611 | 174 |
(rs :: rss, r_set1) |
175 |
} |
|
46661
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
176 |
((List.empty[List[Key]], empty_keys) /: xs)(reachs) |
46611 | 177 |
} |
178 |
||
179 |
/*immediate*/ |
|
180 |
def imm_preds(x: Key): Keys = get_entry(x)._2._1 |
|
181 |
def imm_succs(x: Key): Keys = get_entry(x)._2._2 |
|
182 |
||
183 |
/*transitive*/ |
|
72065
11dc8929832d
clarified order --- proper sorting of requirements;
wenzelm
parents:
71601
diff
changeset
|
184 |
def all_preds_rev(xs: List[Key]): List[Key] = |
11dc8929832d
clarified order --- proper sorting of requirements;
wenzelm
parents:
71601
diff
changeset
|
185 |
reachable(imm_preds, xs, rev = true)._1.flatten.reverse |
46611 | 186 |
def all_preds(xs: List[Key]): List[Key] = reachable(imm_preds, xs)._1.flatten |
187 |
def all_succs(xs: List[Key]): List[Key] = reachable(imm_succs, xs)._1.flatten |
|
188 |
||
46613 | 189 |
/*strongly connected components; see: David King and John Launchbury, |
190 |
"Structuring Depth First Search Algorithms in Haskell"*/ |
|
191 |
def strong_conn: List[List[Key]] = |
|
56372
fadb0fef09d7
more explicit iterator terminology, in accordance to Scala 2.8 library;
wenzelm
parents:
50452
diff
changeset
|
192 |
reachable(imm_preds, all_succs(keys))._1.filterNot(_.isEmpty).reverse |
46613 | 193 |
|
46611 | 194 |
|
195 |
/* minimal and maximal elements */ |
|
196 |
||
197 |
def minimals: List[Key] = |
|
198 |
(List.empty[Key] /: rep) { |
|
199 |
case (ms, (m, (_, (preds, _)))) => if (preds.isEmpty) m :: ms else ms } |
|
200 |
||
201 |
def maximals: List[Key] = |
|
202 |
(List.empty[Key] /: rep) { |
|
203 |
case (ms, (m, (_, (_, succs)))) => if (succs.isEmpty) m :: ms else ms } |
|
204 |
||
205 |
def is_minimal(x: Key): Boolean = imm_preds(x).isEmpty |
|
206 |
def is_maximal(x: Key): Boolean = imm_succs(x).isEmpty |
|
207 |
||
66830 | 208 |
def is_isolated(x: Key): Boolean = is_minimal(x) && is_maximal(x) |
209 |
||
46611 | 210 |
|
46668 | 211 |
/* node operations */ |
46611 | 212 |
|
213 |
def new_node(x: Key, info: A): Graph[Key, A] = |
|
214 |
{ |
|
48648 | 215 |
if (defined(x)) throw new Graph.Duplicate(x) |
46661
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
216 |
else new Graph[Key, A](rep + (x -> (info, (empty_keys, empty_keys)))) |
46611 | 217 |
} |
218 |
||
46613 | 219 |
def default_node(x: Key, info: A): Graph[Key, A] = |
48648 | 220 |
if (defined(x)) this else new_node(x, info) |
46613 | 221 |
|
46661
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
222 |
private def del_adjacent(fst: Boolean, x: Key)(map: SortedMap[Key, Entry], y: Key) |
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
223 |
: SortedMap[Key, Entry] = |
46611 | 224 |
map.get(y) match { |
225 |
case None => map |
|
226 |
case Some((i, (preds, succs))) => |
|
227 |
map + (y -> (i, if (fst) (preds - x, succs) else (preds, succs - x))) |
|
228 |
} |
|
229 |
||
230 |
def del_node(x: Key): Graph[Key, A] = |
|
231 |
{ |
|
232 |
val (preds, succs) = get_entry(x)._2 |
|
233 |
new Graph[Key, A]( |
|
234 |
(((rep - x) /: preds)(del_adjacent(false, x)) /: succs)(del_adjacent(true, x))) |
|
235 |
} |
|
236 |
||
46614
165886a4fe64
clarified Graph.restrict (formerly Graph.subgraph) based on public graph operations;
wenzelm
parents:
46613
diff
changeset
|
237 |
def restrict(pred: Key => Boolean): Graph[Key, A] = |
56372
fadb0fef09d7
more explicit iterator terminology, in accordance to Scala 2.8 library;
wenzelm
parents:
50452
diff
changeset
|
238 |
(this /: iterator){ case (graph, (x, _)) => if (!pred(x)) graph.del_node(x) else graph } |
46614
165886a4fe64
clarified Graph.restrict (formerly Graph.subgraph) based on public graph operations;
wenzelm
parents:
46613
diff
changeset
|
239 |
|
70699 | 240 |
def exclude(pred: Key => Boolean): Graph[Key, A] = restrict(name => !pred(name)) |
241 |
||
46611 | 242 |
|
46668 | 243 |
/* edge operations */ |
46611 | 244 |
|
59259
399506ee38a5
clarified static full_graph vs. dynamic visible_graph;
wenzelm
parents:
59245
diff
changeset
|
245 |
def edges_iterator: Iterator[(Key, Key)] = |
399506ee38a5
clarified static full_graph vs. dynamic visible_graph;
wenzelm
parents:
59245
diff
changeset
|
246 |
for { x <- keys_iterator; y <- imm_succs(x).iterator } yield (x, y) |
399506ee38a5
clarified static full_graph vs. dynamic visible_graph;
wenzelm
parents:
59245
diff
changeset
|
247 |
|
46611 | 248 |
def is_edge(x: Key, y: Key): Boolean = |
48507 | 249 |
defined(x) && defined(y) && imm_succs(x)(y) |
46611 | 250 |
|
251 |
def add_edge(x: Key, y: Key): Graph[Key, A] = |
|
252 |
if (is_edge(x, y)) this |
|
253 |
else |
|
254 |
map_entry(y, { case (i, (preds, succs)) => (i, (preds + x, succs)) }). |
|
255 |
map_entry(x, { case (i, (preds, succs)) => (i, (preds, succs + y)) }) |
|
256 |
||
257 |
def del_edge(x: Key, y: Key): Graph[Key, A] = |
|
258 |
if (is_edge(x, y)) |
|
259 |
map_entry(y, { case (i, (preds, succs)) => (i, (preds - x, succs)) }). |
|
260 |
map_entry(x, { case (i, (preds, succs)) => (i, (preds, succs - y)) }) |
|
261 |
else this |
|
262 |
||
263 |
||
264 |
/* irreducible paths -- Hasse diagram */ |
|
265 |
||
46661
d2ac78ba805e
prefer sorted Map/Set for canonical order of results -- pass ordering via fresh copy of empty;
wenzelm
parents:
46659
diff
changeset
|
266 |
private def irreducible_preds(x_set: Keys, path: List[Key], z: Key): List[Key] = |
46611 | 267 |
{ |
268 |
def red(x: Key)(x1: Key) = is_edge(x, x1) && x1 != z |
|
269 |
@tailrec def irreds(xs0: List[Key], xs1: List[Key]): List[Key] = |
|
270 |
xs0 match { |
|
271 |
case Nil => xs1 |
|
272 |
case x :: xs => |
|
60215 | 273 |
if (!x_set(x) || x == z || path.contains(x) || |
46611 | 274 |
xs.exists(red(x)) || xs1.exists(red(x))) |
275 |
irreds(xs, xs1) |
|
276 |
else irreds(xs, x :: xs1) |
|
277 |
} |
|
278 |
irreds(imm_preds(z).toList, Nil) |
|
279 |
} |
|
280 |
||
281 |
def irreducible_paths(x: Key, y: Key): List[List[Key]] = |
|
282 |
{ |
|
283 |
val (_, x_set) = reachable(imm_succs, List(x)) |
|
284 |
def paths(path: List[Key])(ps: List[List[Key]], z: Key): List[List[Key]] = |
|
285 |
if (x == z) (z :: path) :: ps |
|
286 |
else (ps /: irreducible_preds(x_set, path, z))(paths(z :: path)) |
|
287 |
if ((x == y) && !is_edge(x, x)) List(Nil) else paths(Nil)(Nil, y) |
|
288 |
} |
|
289 |
||
290 |
||
50445
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
291 |
/* transitive closure and reduction */ |
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
292 |
|
50447
2e22cdccdc38
clarified transitive_closure: proper cumulation of transitive steps, which is essential for Warshall-style algorithms;
wenzelm
parents:
50445
diff
changeset
|
293 |
private def transitive_step(z: Key): Graph[Key, A] = |
50445
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
294 |
{ |
50447
2e22cdccdc38
clarified transitive_closure: proper cumulation of transitive steps, which is essential for Warshall-style algorithms;
wenzelm
parents:
50445
diff
changeset
|
295 |
val (preds, succs) = get_entry(z)._2 |
50445
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
296 |
var graph = this |
50447
2e22cdccdc38
clarified transitive_closure: proper cumulation of transitive steps, which is essential for Warshall-style algorithms;
wenzelm
parents:
50445
diff
changeset
|
297 |
for (x <- preds; y <- succs) graph = graph.add_edge(x, y) |
50445
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
298 |
graph |
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
299 |
} |
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
300 |
|
56372
fadb0fef09d7
more explicit iterator terminology, in accordance to Scala 2.8 library;
wenzelm
parents:
50452
diff
changeset
|
301 |
def transitive_closure: Graph[Key, A] = (this /: keys_iterator)(_.transitive_step(_)) |
50447
2e22cdccdc38
clarified transitive_closure: proper cumulation of transitive steps, which is essential for Warshall-style algorithms;
wenzelm
parents:
50445
diff
changeset
|
302 |
|
50445
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
303 |
def transitive_reduction_acyclic: Graph[Key, A] = |
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
304 |
{ |
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
305 |
val trans = this.transitive_closure |
56372
fadb0fef09d7
more explicit iterator terminology, in accordance to Scala 2.8 library;
wenzelm
parents:
50452
diff
changeset
|
306 |
if (trans.iterator.exists({ case (x, (_, (_, succs))) => succs.contains(x) })) |
50452
bfb5964e3041
stateless dockable window for graphview, which is triggered by the active area of the corresponding diagnostic command;
wenzelm
parents:
50447
diff
changeset
|
307 |
error("Cyclic graph") |
50445
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
308 |
|
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
309 |
var graph = this |
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
310 |
for { |
56372
fadb0fef09d7
more explicit iterator terminology, in accordance to Scala 2.8 library;
wenzelm
parents:
50452
diff
changeset
|
311 |
(x, (_, (_, succs))) <- iterator |
50445
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
312 |
y <- succs |
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
313 |
if trans.imm_preds(y).exists(z => trans.is_edge(x, z)) |
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
314 |
} graph = graph.del_edge(x, y) |
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
315 |
graph |
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
316 |
} |
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
317 |
|
68c9a6538c0e
added graph operations for transitive closure and reduction in Scala -- unproven and thus better left out of the kernel-relevant ML module;
wenzelm
parents:
49560
diff
changeset
|
318 |
|
46611 | 319 |
/* maintain acyclic graphs */ |
320 |
||
321 |
def add_edge_acyclic(x: Key, y: Key): Graph[Key, A] = |
|
322 |
if (is_edge(x, y)) this |
|
323 |
else { |
|
324 |
irreducible_paths(y, x) match { |
|
325 |
case Nil => add_edge(x, y) |
|
326 |
case cycles => throw new Graph.Cycles(cycles.map(x :: _)) |
|
327 |
} |
|
328 |
} |
|
329 |
||
48348 | 330 |
def add_deps_acyclic(y: Key, xs: List[Key]): Graph[Key, A] = |
46611 | 331 |
(this /: xs)(_.add_edge_acyclic(_, y)) |
332 |
||
333 |
def topological_order: List[Key] = all_succs(minimals) |
|
334 |
} |