COIN-OR::LEMON - Graph Library

source: lemon-main/test/graph_test.cc @ 1163:db1d342a1087

Last change on this file since 1163:db1d342a1087 was 1092:dceba191c00d, checked in by Alpar Juttner <alpar@…>, 11 years ago

Apply unify-sources.sh to the source tree

File size: 15.0 KB
RevLine 
[209]1/* -*- mode: C++; indent-tabs-mode: nil; -*-
[57]2 *
[209]3 * This file is a part of LEMON, a generic C++ optimization library.
[57]4 *
[1092]5 * Copyright (C) 2003-2013
[57]6 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
7 * (Egervary Research Group on Combinatorial Optimization, EGRES).
8 *
9 * Permission to use, modify and distribute this software is granted
10 * provided that this copyright notice appears in all copies. For
11 * precise terms see the accompanying LICENSE file.
12 *
13 * This software is provided "AS IS" with no warranty of any kind,
14 * express or implied, and with no claim as to its suitability for any
15 * purpose.
16 *
17 */
18
19#include <lemon/concepts/graph.h>
20#include <lemon/list_graph.h>
[109]21#include <lemon/smart_graph.h>
[353]22#include <lemon/full_graph.h>
[334]23#include <lemon/grid_graph.h>
[365]24#include <lemon/hypercube_graph.h>
[57]25
26#include "test_tools.h"
[171]27#include "graph_test.h"
[57]28
29using namespace lemon;
30using namespace lemon::concepts;
31
[228]32template <class Graph>
[374]33void checkGraphBuild() {
[228]34  TEMPLATE_GRAPH_TYPEDEFS(Graph);
35
36  Graph G;
37  checkGraphNodeList(G, 0);
38  checkGraphEdgeList(G, 0);
[374]39  checkGraphArcList(G, 0);
[228]40
[736]41  G.reserveNode(3);
42  G.reserveEdge(3);
43
[228]44  Node
45    n1 = G.addNode(),
46    n2 = G.addNode(),
47    n3 = G.addNode();
48  checkGraphNodeList(G, 3);
49  checkGraphEdgeList(G, 0);
[374]50  checkGraphArcList(G, 0);
[228]51
52  Edge e1 = G.addEdge(n1, n2);
53  check((G.u(e1) == n1 && G.v(e1) == n2) || (G.u(e1) == n2 && G.v(e1) == n1),
54        "Wrong edge");
[374]55
[228]56  checkGraphNodeList(G, 3);
[374]57  checkGraphEdgeList(G, 1);
[228]58  checkGraphArcList(G, 2);
59
[374]60  checkGraphIncEdgeArcLists(G, n1, 1);
61  checkGraphIncEdgeArcLists(G, n2, 1);
62  checkGraphIncEdgeArcLists(G, n3, 0);
[228]63
[374]64  checkGraphConEdgeList(G, 1);
65  checkGraphConArcList(G, 2);
[228]66
[374]67  Edge e2 = G.addEdge(n2, n1),
68       e3 = G.addEdge(n2, n3);
[1083]69  ::lemon::ignore_unused_variable_warning(e2,e3);
[228]70
[374]71  checkGraphNodeList(G, 3);
72  checkGraphEdgeList(G, 3);
73  checkGraphArcList(G, 6);
[228]74
[374]75  checkGraphIncEdgeArcLists(G, n1, 2);
76  checkGraphIncEdgeArcLists(G, n2, 3);
77  checkGraphIncEdgeArcLists(G, n3, 1);
[228]78
[374]79  checkGraphConEdgeList(G, 3);
[228]80  checkGraphConArcList(G, 6);
81
82  checkArcDirections(G);
83
84  checkNodeIds(G);
85  checkArcIds(G);
86  checkEdgeIds(G);
87  checkGraphNodeMap(G);
88  checkGraphArcMap(G);
89  checkGraphEdgeMap(G);
90}
91
[374]92template <class Graph>
93void checkGraphAlter() {
94  TEMPLATE_GRAPH_TYPEDEFS(Graph);
95
96  Graph G;
97  Node n1 = G.addNode(), n2 = G.addNode(),
98       n3 = G.addNode(), n4 = G.addNode();
99  Edge e1 = G.addEdge(n1, n2), e2 = G.addEdge(n2, n1),
100       e3 = G.addEdge(n2, n3), e4 = G.addEdge(n1, n4),
101       e5 = G.addEdge(n4, n3);
[1083]102  ::lemon::ignore_unused_variable_warning(e1,e3,e4,e5);
[374]103
104  checkGraphNodeList(G, 4);
105  checkGraphEdgeList(G, 5);
106  checkGraphArcList(G, 10);
107
108  // Check changeU() and changeV()
109  if (G.u(e2) == n2) {
110    G.changeU(e2, n3);
111  } else {
112    G.changeV(e2, n3);
113  }
114
115  checkGraphNodeList(G, 4);
116  checkGraphEdgeList(G, 5);
117  checkGraphArcList(G, 10);
118
119  checkGraphIncEdgeArcLists(G, n1, 3);
120  checkGraphIncEdgeArcLists(G, n2, 2);
121  checkGraphIncEdgeArcLists(G, n3, 3);
122  checkGraphIncEdgeArcLists(G, n4, 2);
123
124  checkGraphConEdgeList(G, 5);
125  checkGraphConArcList(G, 10);
126
127  if (G.u(e2) == n1) {
128    G.changeU(e2, n2);
129  } else {
130    G.changeV(e2, n2);
131  }
132
133  checkGraphNodeList(G, 4);
134  checkGraphEdgeList(G, 5);
135  checkGraphArcList(G, 10);
136
137  checkGraphIncEdgeArcLists(G, n1, 2);
138  checkGraphIncEdgeArcLists(G, n2, 3);
139  checkGraphIncEdgeArcLists(G, n3, 3);
140  checkGraphIncEdgeArcLists(G, n4, 2);
141
142  checkGraphConEdgeList(G, 5);
143  checkGraphConArcList(G, 10);
144
145  // Check contract()
146  G.contract(n1, n4, false);
147
148  checkGraphNodeList(G, 3);
149  checkGraphEdgeList(G, 5);
150  checkGraphArcList(G, 10);
151
152  checkGraphIncEdgeArcLists(G, n1, 4);
153  checkGraphIncEdgeArcLists(G, n2, 3);
154  checkGraphIncEdgeArcLists(G, n3, 3);
155
156  checkGraphConEdgeList(G, 5);
157  checkGraphConArcList(G, 10);
158
159  G.contract(n2, n3);
160
161  checkGraphNodeList(G, 2);
162  checkGraphEdgeList(G, 3);
163  checkGraphArcList(G, 6);
164
165  checkGraphIncEdgeArcLists(G, n1, 4);
166  checkGraphIncEdgeArcLists(G, n2, 2);
167
168  checkGraphConEdgeList(G, 3);
169  checkGraphConArcList(G, 6);
170}
171
172template <class Graph>
173void checkGraphErase() {
174  TEMPLATE_GRAPH_TYPEDEFS(Graph);
175
176  Graph G;
177  Node n1 = G.addNode(), n2 = G.addNode(),
178       n3 = G.addNode(), n4 = G.addNode();
179  Edge e1 = G.addEdge(n1, n2), e2 = G.addEdge(n2, n1),
180       e3 = G.addEdge(n2, n3), e4 = G.addEdge(n1, n4),
181       e5 = G.addEdge(n4, n3);
[1083]182  ::lemon::ignore_unused_variable_warning(e1,e3,e4,e5);
[374]183
184  // Check edge deletion
185  G.erase(e2);
186
187  checkGraphNodeList(G, 4);
188  checkGraphEdgeList(G, 4);
189  checkGraphArcList(G, 8);
190
191  checkGraphIncEdgeArcLists(G, n1, 2);
192  checkGraphIncEdgeArcLists(G, n2, 2);
193  checkGraphIncEdgeArcLists(G, n3, 2);
194  checkGraphIncEdgeArcLists(G, n4, 2);
195
196  checkGraphConEdgeList(G, 4);
197  checkGraphConArcList(G, 8);
198
199  // Check node deletion
200  G.erase(n3);
201
202  checkGraphNodeList(G, 3);
203  checkGraphEdgeList(G, 2);
204  checkGraphArcList(G, 4);
205
206  checkGraphIncEdgeArcLists(G, n1, 2);
207  checkGraphIncEdgeArcLists(G, n2, 1);
208  checkGraphIncEdgeArcLists(G, n4, 1);
209
210  checkGraphConEdgeList(G, 2);
211  checkGraphConArcList(G, 4);
212}
213
214
215template <class Graph>
216void checkGraphSnapshot() {
217  TEMPLATE_GRAPH_TYPEDEFS(Graph);
218
219  Graph G;
220  Node n1 = G.addNode(), n2 = G.addNode(), n3 = G.addNode();
221  Edge e1 = G.addEdge(n1, n2), e2 = G.addEdge(n2, n1),
222       e3 = G.addEdge(n2, n3);
[1083]223  ::lemon::ignore_unused_variable_warning(e1,e2,e3);
[374]224
225  checkGraphNodeList(G, 3);
226  checkGraphEdgeList(G, 3);
227  checkGraphArcList(G, 6);
228
229  typename Graph::Snapshot snapshot(G);
230
231  Node n = G.addNode();
232  G.addEdge(n3, n);
233  G.addEdge(n, n3);
234  G.addEdge(n3, n2);
235
236  checkGraphNodeList(G, 4);
237  checkGraphEdgeList(G, 6);
238  checkGraphArcList(G, 12);
239
240  snapshot.restore();
241
242  checkGraphNodeList(G, 3);
243  checkGraphEdgeList(G, 3);
244  checkGraphArcList(G, 6);
245
246  checkGraphIncEdgeArcLists(G, n1, 2);
247  checkGraphIncEdgeArcLists(G, n2, 3);
248  checkGraphIncEdgeArcLists(G, n3, 1);
249
250  checkGraphConEdgeList(G, 3);
251  checkGraphConArcList(G, 6);
252
253  checkNodeIds(G);
254  checkEdgeIds(G);
255  checkArcIds(G);
256  checkGraphNodeMap(G);
257  checkGraphEdgeMap(G);
258  checkGraphArcMap(G);
259
260  G.addNode();
261  snapshot.save(G);
262
263  G.addEdge(G.addNode(), G.addNode());
264
265  snapshot.restore();
[740]266  snapshot.save(G);
267
268  checkGraphNodeList(G, 4);
269  checkGraphEdgeList(G, 3);
270  checkGraphArcList(G, 6);
[877]271
[740]272  G.addEdge(G.addNode(), G.addNode());
273
274  snapshot.restore();
[374]275
276  checkGraphNodeList(G, 4);
277  checkGraphEdgeList(G, 3);
278  checkGraphArcList(G, 6);
279}
280
[353]281void checkFullGraph(int num) {
282  typedef FullGraph Graph;
283  GRAPH_TYPEDEFS(Graph);
284
285  Graph G(num);
[737]286  check(G.nodeNum() == num && G.edgeNum() == num * (num - 1) / 2,
287        "Wrong size");
288
289  G.resize(num);
290  check(G.nodeNum() == num && G.edgeNum() == num * (num - 1) / 2,
291        "Wrong size");
292
[353]293  checkGraphNodeList(G, num);
294  checkGraphEdgeList(G, num * (num - 1) / 2);
295
296  for (NodeIt n(G); n != INVALID; ++n) {
[365]297    checkGraphOutArcList(G, n, num - 1);
298    checkGraphInArcList(G, n, num - 1);
299    checkGraphIncEdgeList(G, n, num - 1);
[353]300  }
301
302  checkGraphConArcList(G, num * (num - 1));
303  checkGraphConEdgeList(G, num * (num - 1) / 2);
304
305  checkArcDirections(G);
306
307  checkNodeIds(G);
308  checkArcIds(G);
309  checkEdgeIds(G);
310  checkGraphNodeMap(G);
311  checkGraphArcMap(G);
312  checkGraphEdgeMap(G);
313
[365]314
[353]315  for (int i = 0; i < G.nodeNum(); ++i) {
316    check(G.index(G(i)) == i, "Wrong index");
317  }
318
319  for (NodeIt u(G); u != INVALID; ++u) {
320    for (NodeIt v(G); v != INVALID; ++v) {
321      Edge e = G.edge(u, v);
322      Arc a = G.arc(u, v);
323      if (u == v) {
324        check(e == INVALID, "Wrong edge lookup");
325        check(a == INVALID, "Wrong arc lookup");
326      } else {
327        check((G.u(e) == u && G.v(e) == v) ||
328              (G.u(e) == v && G.v(e) == u), "Wrong edge lookup");
329        check(G.source(a) == u && G.target(a) == v, "Wrong arc lookup");
330      }
331    }
332  }
333}
334
[228]335void checkConcepts() {
[171]336  { // Checking graph components
[57]337    checkConcept<BaseGraphComponent, BaseGraphComponent >();
338
[209]339    checkConcept<IDableGraphComponent<>,
[57]340      IDableGraphComponent<> >();
341
[209]342    checkConcept<IterableGraphComponent<>,
[57]343      IterableGraphComponent<> >();
344
[209]345    checkConcept<MappableGraphComponent<>,
[57]346      MappableGraphComponent<> >();
347  }
[171]348  { // Checking skeleton graph
349    checkConcept<Graph, Graph>();
[57]350  }
[171]351  { // Checking ListGraph
352    checkConcept<Graph, ListGraph>();
353    checkConcept<AlterableGraphComponent<>, ListGraph>();
354    checkConcept<ExtendableGraphComponent<>, ListGraph>();
355    checkConcept<ClearableGraphComponent<>, ListGraph>();
356    checkConcept<ErasableGraphComponent<>, ListGraph>();
357  }
358  { // Checking SmartGraph
359    checkConcept<Graph, SmartGraph>();
360    checkConcept<AlterableGraphComponent<>, SmartGraph>();
361    checkConcept<ExtendableGraphComponent<>, SmartGraph>();
362    checkConcept<ClearableGraphComponent<>, SmartGraph>();
363  }
[353]364  { // Checking FullGraph
365    checkConcept<Graph, FullGraph>();
366  }
[334]367  { // Checking GridGraph
368    checkConcept<Graph, GridGraph>();
369  }
[365]370  { // Checking HypercubeGraph
371    checkConcept<Graph, HypercubeGraph>();
372  }
[57]373}
374
375template <typename Graph>
[228]376void checkGraphValidity() {
[149]377  TEMPLATE_GRAPH_TYPEDEFS(Graph);
[57]378  Graph g;
379
380  Node
381    n1 = g.addNode(),
382    n2 = g.addNode(),
383    n3 = g.addNode();
384
385  Edge
386    e1 = g.addEdge(n1, n2),
387    e2 = g.addEdge(n2, n3);
[1083]388  ::lemon::ignore_unused_variable_warning(e2);
[57]389
[171]390  check(g.valid(n1), "Wrong validity check");
391  check(g.valid(e1), "Wrong validity check");
392  check(g.valid(g.direct(e1, true)), "Wrong validity check");
393
394  check(!g.valid(g.nodeFromId(-1)), "Wrong validity check");
395  check(!g.valid(g.edgeFromId(-1)), "Wrong validity check");
396  check(!g.valid(g.arcFromId(-1)), "Wrong validity check");
[57]397}
398
[149]399template <typename Graph>
[228]400void checkGraphValidityErase() {
[149]401  TEMPLATE_GRAPH_TYPEDEFS(Graph);
402  Graph g;
403
404  Node
405    n1 = g.addNode(),
406    n2 = g.addNode(),
407    n3 = g.addNode();
408
409  Edge
410    e1 = g.addEdge(n1, n2),
411    e2 = g.addEdge(n2, n3);
412
[171]413  check(g.valid(n1), "Wrong validity check");
414  check(g.valid(e1), "Wrong validity check");
415  check(g.valid(g.direct(e1, true)), "Wrong validity check");
[149]416
417  g.erase(n1);
418
[171]419  check(!g.valid(n1), "Wrong validity check");
420  check(g.valid(n2), "Wrong validity check");
421  check(g.valid(n3), "Wrong validity check");
422  check(!g.valid(e1), "Wrong validity check");
423  check(g.valid(e2), "Wrong validity check");
[149]424
[171]425  check(!g.valid(g.nodeFromId(-1)), "Wrong validity check");
426  check(!g.valid(g.edgeFromId(-1)), "Wrong validity check");
427  check(!g.valid(g.arcFromId(-1)), "Wrong validity check");
[149]428}
429
[335]430void checkGridGraph(int width, int height) {
431  typedef GridGraph Graph;
432  GRAPH_TYPEDEFS(Graph);
433  Graph G(width, height);
[57]434
[336]435  check(G.width() == width, "Wrong column number");
436  check(G.height() == height, "Wrong row number");
[209]437
[737]438  G.resize(width, height);
439  check(G.width() == width, "Wrong column number");
440  check(G.height() == height, "Wrong row number");
441
[335]442  for (int i = 0; i < width; ++i) {
443    for (int j = 0; j < height; ++j) {
444      check(G.col(G(i, j)) == i, "Wrong column");
445      check(G.row(G(i, j)) == j, "Wrong row");
446      check(G.pos(G(i, j)).x == i, "Wrong column");
447      check(G.pos(G(i, j)).y == j, "Wrong row");
[334]448    }
449  }
[57]450
[335]451  for (int j = 0; j < height; ++j) {
452    for (int i = 0; i < width - 1; ++i) {
453      check(G.source(G.right(G(i, j))) == G(i, j), "Wrong right");
454      check(G.target(G.right(G(i, j))) == G(i + 1, j), "Wrong right");
[334]455    }
[335]456    check(G.right(G(width - 1, j)) == INVALID, "Wrong right");
[334]457  }
[57]458
[335]459  for (int j = 0; j < height; ++j) {
460    for (int i = 1; i < width; ++i) {
461      check(G.source(G.left(G(i, j))) == G(i, j), "Wrong left");
462      check(G.target(G.left(G(i, j))) == G(i - 1, j), "Wrong left");
[334]463    }
[335]464    check(G.left(G(0, j)) == INVALID, "Wrong left");
[334]465  }
[57]466
[335]467  for (int i = 0; i < width; ++i) {
468    for (int j = 0; j < height - 1; ++j) {
469      check(G.source(G.up(G(i, j))) == G(i, j), "Wrong up");
470      check(G.target(G.up(G(i, j))) == G(i, j + 1), "Wrong up");
[334]471    }
[335]472    check(G.up(G(i, height - 1)) == INVALID, "Wrong up");
[334]473  }
[228]474
[335]475  for (int i = 0; i < width; ++i) {
476    for (int j = 1; j < height; ++j) {
477      check(G.source(G.down(G(i, j))) == G(i, j), "Wrong down");
478      check(G.target(G.down(G(i, j))) == G(i, j - 1), "Wrong down");
[334]479    }
[335]480    check(G.down(G(i, 0)) == INVALID, "Wrong down");
[334]481  }
482
[335]483  checkGraphNodeList(G, width * height);
484  checkGraphEdgeList(G, width * (height - 1) + (width - 1) * height);
485  checkGraphArcList(G, 2 * (width * (height - 1) + (width - 1) * height));
[334]486
[335]487  for (NodeIt n(G); n != INVALID; ++n) {
488    int nb = 4;
489    if (G.col(n) == 0) --nb;
490    if (G.col(n) == width - 1) --nb;
491    if (G.row(n) == 0) --nb;
492    if (G.row(n) == height - 1) --nb;
[334]493
[335]494    checkGraphOutArcList(G, n, nb);
495    checkGraphInArcList(G, n, nb);
496    checkGraphIncEdgeList(G, n, nb);
497  }
[334]498
[335]499  checkArcDirections(G);
[334]500
[335]501  checkGraphConArcList(G, 2 * (width * (height - 1) + (width - 1) * height));
502  checkGraphConEdgeList(G, width * (height - 1) + (width - 1) * height);
[334]503
[335]504  checkNodeIds(G);
505  checkArcIds(G);
506  checkEdgeIds(G);
507  checkGraphNodeMap(G);
508  checkGraphArcMap(G);
509  checkGraphEdgeMap(G);
[334]510
511}
[228]512
[365]513void checkHypercubeGraph(int dim) {
514  GRAPH_TYPEDEFS(HypercubeGraph);
515
516  HypercubeGraph G(dim);
[737]517  check(G.dimension() == dim, "Wrong dimension");
518
519  G.resize(dim);
520  check(G.dimension() == dim, "Wrong dimension");
[877]521
[365]522  checkGraphNodeList(G, 1 << dim);
[372]523  checkGraphEdgeList(G, dim * (1 << (dim-1)));
[365]524  checkGraphArcList(G, dim * (1 << dim));
525
526  Node n = G.nodeFromId(dim);
[1083]527  ::lemon::ignore_unused_variable_warning(n);
[365]528
529  for (NodeIt n(G); n != INVALID; ++n) {
530    checkGraphIncEdgeList(G, n, dim);
531    for (IncEdgeIt e(G, n); e != INVALID; ++e) {
532      check( (G.u(e) == n &&
[372]533              G.id(G.v(e)) == (G.id(n) ^ (1 << G.dimension(e)))) ||
[365]534             (G.v(e) == n &&
[372]535              G.id(G.u(e)) == (G.id(n) ^ (1 << G.dimension(e)))),
[365]536             "Wrong edge or wrong dimension");
537    }
538
539    checkGraphOutArcList(G, n, dim);
540    for (OutArcIt a(G, n); a != INVALID; ++a) {
541      check(G.source(a) == n &&
[372]542            G.id(G.target(a)) == (G.id(n) ^ (1 << G.dimension(a))),
[365]543            "Wrong arc or wrong dimension");
544    }
545
546    checkGraphInArcList(G, n, dim);
547    for (InArcIt a(G, n); a != INVALID; ++a) {
548      check(G.target(a) == n &&
[372]549            G.id(G.source(a)) == (G.id(n) ^ (1 << G.dimension(a))),
[365]550            "Wrong arc or wrong dimension");
551    }
552  }
553
554  checkGraphConArcList(G, (1 << dim) * dim);
[372]555  checkGraphConEdgeList(G, dim * (1 << (dim-1)));
[365]556
557  checkArcDirections(G);
558
559  checkNodeIds(G);
560  checkArcIds(G);
561  checkEdgeIds(G);
562  checkGraphNodeMap(G);
563  checkGraphArcMap(G);
564  checkGraphEdgeMap(G);
565}
[57]566
[228]567void checkGraphs() {
[171]568  { // Checking ListGraph
[374]569    checkGraphBuild<ListGraph>();
570    checkGraphAlter<ListGraph>();
571    checkGraphErase<ListGraph>();
572    checkGraphSnapshot<ListGraph>();
[228]573    checkGraphValidityErase<ListGraph>();
[171]574  }
575  { // Checking SmartGraph
[374]576    checkGraphBuild<SmartGraph>();
577    checkGraphSnapshot<SmartGraph>();
[228]578    checkGraphValidity<SmartGraph>();
[171]579  }
[365]580  { // Checking FullGraph
[353]581    checkFullGraph(7);
582    checkFullGraph(8);
583  }
[334]584  { // Checking GridGraph
[335]585    checkGridGraph(5, 8);
586    checkGridGraph(8, 5);
587    checkGridGraph(5, 5);
588    checkGridGraph(0, 0);
589    checkGridGraph(1, 1);
[334]590  }
[365]591  { // Checking HypercubeGraph
592    checkHypercubeGraph(1);
593    checkHypercubeGraph(2);
594    checkHypercubeGraph(3);
595    checkHypercubeGraph(4);
596  }
[171]597}
598
[57]599int main() {
[228]600  checkConcepts();
601  checkGraphs();
[57]602  return 0;
603}
Note: See TracBrowser for help on using the repository browser.