diff --git a/alib2algo/src/graph/spanningtree/Edmonds.cpp b/alib2algo/src/graph/spanningtree/Edmonds.cpp new file mode 100644 index 0000000000000000000000000000000000000000..ac51a91e60dd615228daa38cd825df62ea1f4ceb --- /dev/null +++ b/alib2algo/src/graph/spanningtree/Edmonds.cpp @@ -0,0 +1,254 @@ +/* + * Edmonds.cpp + * + * Created on: Apr 7, 2016 + * Author: Jan Broz + */ + +#include "Edmonds.h" + +#include <vector> +#include <list> +#include <queue> +#include <climits> +#include <stdexcept> + +#include <graph/GraphClasses.h> +#include "../datastructs/BinomialHeap.h" +#include "../datastructs/FibonacciHeap.h" +#include "../datastructs/Components.h" +#include "../datastructs/Array.h" + +#include <exception/AlibException.h> + +namespace graph { + +namespace spanningtree { + +using namespace std; + +//==================================================================================================== +// data structures + +/// wrapper for directed edge with additional info +struct Edge { + DirectedEdge e; + uint from; + uint to; + int weight; + int origWeight; + Edge * fParent; + vector<Edge*> fChildren; + bool deleted; + + Edge( const DirectedEdge & edge, uint from, uint to, int weight ) + : e(edge), from(from), to(to), weight(weight), origWeight(weight), fParent(NULL), deleted(false) {} +}; + +static int compareEdges( Edge * const & e1, Edge * const & e2 ) +{ + return e2->weight - e1->weight; +} + +class EdgeQueue : public BinomialHeap<Edge*> { + + public: + + EdgeQueue() : BinomialHeap( compareEdges ), offset( 0 ) {} + ~EdgeQueue() {} + + void decreaseAll( int amount ) { offset -= amount; } + + Edge * extractMax() + { + if (size() == 0) + return NULL; + Edge * edge = BinomialHeap<Edge*>::extractMax(); + edge->weight += offset; + return edge; + } + + protected: + + int offset; + +}; + +template< typename Type > +void vector_remove( vector<Type> & vec, Type elem ) +{ + for (auto it = vec.begin(); it != vec.end(); it++) { + if (*it == elem) { + vec.erase( it ); + break; + } + } +} + +//==================================================================================================== +// algorithm + +static AdjacencyListDirectedGraph edmonds_impl( const DirectedGraph & graph ) +{ + AdjacencyListDirectedGraph spanningTree; // spanning tree + + uint nodeCnt = graph.nodeCount(); + uint edgeCnt = graph.edgeCount(); + + list<Edge> edges; // just to ease deallocation + queue<uint> roots; ///< silne komponenty, ktere jsou koreny v aktualnim grafu (nevede do nich zadny uzel) + Array<EdgeQueue> queues( nodeCnt ); ///< vstupni hrany do danych uzlu (razene dle priority) + Array<Edge*> enter( nodeCnt ); ///< vstupni hrany do silnych komponent + Array<vector<Edge*>> h( nodeCnt ); ///< hrany, ktere mohou byt v kostre + vector<uint> rset; ///< rooti vyslednych stromu Tarjanovy implementace + Components2 strong( nodeCnt ); ///< silne komponenty (reprezentanti) + Components2 weak( nodeCnt ); ///< slabe komponenty (reprezentanti) + Array<uint> min( nodeCnt ); ///< urcuje finalni korenove uzly orientovanych koster + vector<Edge*> fRoots; ///< koreny lesa F + Array<vector<Edge*>> cycles( nodeCnt ); ///< nalezene cykly + Array<Edge*> lambda( nodeCnt ); ///< listy lesa F + + // this algorithm works on number identification of nodes and edges + std::vector<Node> id2node( nodeCnt ); + std::vector<DirectedEdge> id2edge( edgeCnt ); + std::unordered_map<Node,uint> node2id; + std::unordered_map<DirectedEdge,uint> edge2id; + uint id = 0; + for (const Node & node : graph.getNodes()) { + id2node[ id ] = node; + node2id[ node ] = id; + id++; + } +/* id = 0; + for (const DirectedEdge & edge : graph.getEdges()) { + id2edge[ id ] = edge; + edge2id[ edge ] = id; + id++; + } +*/ + // fill up data structures + for (uint nodeID = 0; nodeID < nodeCnt; nodeID++) { + roots.push( nodeID ); + enter[ nodeID ] = NULL; + min[ nodeID ] = nodeID; + strong.MakeSet( nodeID ); + weak.MakeSet( nodeID ); + lambda[ nodeID ] = NULL; + } + for (const DirectedEdge & edge : graph.getEdges()) { + int weight; + try { + weight = graph.getEdgeValue( edge ); + } catch (const std::out_of_range & ex) { + throw exception::AlibException("Edmonds: an edge is missing a capacity"); + } + edges.emplace_back( edge, node2id[ edge.getFromNode() ], node2id[ edge.getToNode() ], weight ); + queues[ node2id[edge.getToNode()] ].insert( &edges.back() ); + } + + while (roots.size() != 0) { + uint root = roots.front(); roots.pop(); + Edge * edge = queues[ root ].extractMax(); + if (edge != NULL) { + if (cycles[ strong.Find( root ) ].size() <= 1) { + lambda[ root ] = edge; + } else if (strong.Find( edge->from ) != strong.Find( edge->to )) { + for (Edge * e : cycles[ strong.Find( root ) ]) { + vector_remove( fRoots, e ); //je v cyklu, nemuze byt rootem + edge->fChildren.push_back( e ); + e->fParent = edge; + } + } + } + if (edge == NULL) { + rset.push_back( root ); //pak uzel nema vstupni hranu (je root) + } else if (strong.Find( edge->from ) == root) { //nalezli jsme hranu v ramci silne komponenty + roots.push( root ); //uzel vratimem do fronty + } else { + h[ edge->from ].push_back( edge ); //tato hrana je adeptem na hranu MST + fRoots.push_back( edge ); //tato hrana je korenem v lese F + if (weak.Find( edge->from ) != weak.Find( edge->to )) { //pocatecni a koncovy uzel jsou v ruznych slabych komponentach + weak.Union( weak.Find( edge->from ), weak.Find( edge->to )); + enter[ root ] = edge; //edge je vstupni hranou do root + } else { //pocatecni a koncovy uzel jsou ve stejne slabe komponente + Edge * cycleEdge = edge; + int maxVal = INT_MIN; + int vertex = INT_MIN; + + vector<Edge*> & cycle = cycles[ strong.Find( root ) ]; + cycle.clear(); + while (cycleEdge != NULL) { //hledame hranu v cyklu s maximalni vahou (ktera nejvice poskodi cyklus) + cycle.push_back( cycleEdge ); + if (cycleEdge->weight > maxVal) { + maxVal = cycleEdge->weight; + vertex = strong.Find( cycleEdge->to ); + } + cycleEdge = enter[ strong.Find( cycleEdge->from ) ]; + } + queues[ root ].decreaseAll( edge->weight - maxVal ); + min[ root ] = min[ vertex ]; + + //collapse cycle + cycleEdge = enter[ strong.Find( edge->from ) ]; + while (cycleEdge != NULL) { + queues[ strong.Find( cycleEdge->to ) ].decreaseAll( cycleEdge->weight - maxVal ); //pridame prichozim hranam hodnotu, ktera znaci, o kolik se vypustenim dane cyklove hrany a pridanim dane necyklove zhorsi reseni + queues[ root ].mergeWith( std::move( queues[ strong.Find( cycleEdge->to ) ] ) ); + strong.Union( root, strong.Find( cycleEdge->to ) ); + cycleEdge = enter[ strong.Find( cycleEdge->from ) ]; + } + roots.push( root ); + } + } + } + + + vector<Edge*> b; + vector<uint> r; + uint fRootIndex = 0; + + while (!rset.empty() || fRootIndex < fRoots.size()) { + uint v = 0; + if (!rset.empty()) { + v = min[ rset[0] ]; rset.erase( rset.begin() ); + r.push_back(v); + } else { + Edge * e = fRoots[ fRootIndex ]; + if (e->deleted) { + fRootIndex++; + continue; + } + v = e->to; + b.push_back(e); + fRootIndex++; + } + Edge * curr = lambda[ v ]; + while (curr != NULL && !curr->deleted) { + curr->deleted = true; + fRoots.insert( fRoots.begin(), curr->fChildren.begin(), curr->fChildren.end() ); + curr = curr->fParent; + } + } + + for (const Node & node : graph.getNodes()) + spanningTree.addNode( node ); + for (Edge * edge : b) + spanningTree.addEdge( edge->e, edge->origWeight ); + + return spanningTree; +} + +Graph Edmonds::edmonds( const Graph & graph ) +{ + return getInstance().dispatch( graph.getData() ); +} + +AdjacencyListDirectedGraph Edmonds::edmonds( const DirectedGraph & graph ) +{ + return edmonds_impl( graph ); +} + +auto EdmondsDirectedGraph = Edmonds::RegistratorWrapper< graph::AdjacencyListDirectedGraph, graph::DirectedGraph >( Edmonds::getInstance(), Edmonds::edmonds ); + +} // spanningtree + +} // graph diff --git a/alib2algo/src/graph/spanningtree/Edmonds.h b/alib2algo/src/graph/spanningtree/Edmonds.h new file mode 100644 index 0000000000000000000000000000000000000000..6a8f5b9f43086583077c2d13d2986a01264d5a27 --- /dev/null +++ b/alib2algo/src/graph/spanningtree/Edmonds.h @@ -0,0 +1,37 @@ +/* + * Edmonds.h + * + * Created on: Apr 7, 2016 + * Author: Jan Broz + */ + +#ifndef GRAPH_EDMONDS +#define GRAPH_EDMONDS + +#include <core/multipleDispatch.hpp> +#include <graph/GraphClasses.h> + +namespace graph { + +namespace spanningtree { + +class Edmonds : public std::SingleDispatch< graph::Graph, graph::GraphBase > { + + public: + + static Graph edmonds( const Graph & graph ); + + static AdjacencyListDirectedGraph edmonds( const DirectedGraph & graph ); + + static Edmonds & getInstance() { + static Edmonds res; + return res; + } + +}; + +} // spanningtree + +} // graph + +#endif // GRAPH_EDMONDS diff --git a/alib2algo/src/graph/spanningtree/Kruskal.cpp b/alib2algo/src/graph/spanningtree/Kruskal.cpp new file mode 100644 index 0000000000000000000000000000000000000000..426db198bed96c2385ed802ebc438a39de926321 --- /dev/null +++ b/alib2algo/src/graph/spanningtree/Kruskal.cpp @@ -0,0 +1,91 @@ +/* + * Kruskal.cpp + * + * Created on: Mar 26, 2016 + * Author: Jan Broz + */ + +#include "Kruskal.h" + +#include <vector> +#include <unordered_map> +#include <algorithm> +#include <stdexcept> + +#include "../datastructs/Components.h" +#include <exception/AlibException.h> + +namespace graph { + +namespace spanningtree { + +/// wrapper for edges containing weight +struct WeightedEdge { + UndirectedEdge edge; + int weight; + WeightedEdge( const UndirectedEdge & edge, int weight ) : edge(edge), weight(weight) {} +}; + +bool isWeightSmaller( const WeightedEdge & e1, const WeightedEdge & e2) +{ + return e1.weight < e2.weight; +} + +/// converts edges to wrappers with assigned weight +static std::vector<WeightedEdge> getEdges( const UndirectedGraph & graph ) +{ + std::vector<WeightedEdge> edges; + int weight; + + for (const UndirectedEdge & edge : graph.getEdges()) { + try { + weight = graph.getEdgeValue( edge ); + } catch (const std::out_of_range & ex) { // when this edge has no value assigned + throw exception::AlibException("Kruskal: an edge is missing a weight"); + } + edges.emplace_back( edge, weight ); + } + + return edges; +} + +static AdjacencyListUndirectedGraph kruskal_impl( const UndirectedGraph & graph ) +{ + AdjacencyListUndirectedGraph res; // spanning tree + std::vector<WeightedEdge> edges; // edges with weights + Components comps; // components of the emerging spanning tree + + for (const Node & node : graph.getNodes()) { + res.addNode( node ); + comps.MakeSet( node ); + } + + edges = getEdges( graph ); + // sort edges by increasing weight + std::sort( edges.begin(), edges.end(), isWeightSmaller ); + + for (const WeightedEdge & wedge : edges) { // little easter egg here :D + if (comps.Find( wedge.edge.getFirstNode() ) != comps.Find( wedge.edge.getSecondNode() )) { // if both nodes of the edge are in same set, it would create cycle + comps.Union( wedge.edge.getFirstNode(), wedge.edge.getSecondNode() ); + res.addEdge( wedge.edge, wedge.weight ); + } + } + + return res; +} + +Graph Kruskal::kruskal( const Graph & graph ) +{ + return getInstance().dispatch( graph.getData() ); +} + +AdjacencyListUndirectedGraph Kruskal::kruskal( const UndirectedGraph & graph ) +{ + return kruskal_impl( graph ); +} + +auto KruskalUndirectedGraph = Kruskal::RegistratorWrapper< graph::AdjacencyListUndirectedGraph, graph::UndirectedGraph >( Kruskal::getInstance(), Kruskal::kruskal ); + +} // namespace spanningtree + +} // namespace graph diff --git a/alib2algo/src/graph/spanningtree/Kruskal.h b/alib2algo/src/graph/spanningtree/Kruskal.h new file mode 100644 index 0000000000000000000000000000000000000000..22609778b89118223f836be83cabba29f132dd07 --- /dev/null +++ b/alib2algo/src/graph/spanningtree/Kruskal.h @@ -0,0 +1,37 @@ +/* + * Kruskal.h + * + * Created on: Mar 26, 2016 + * Author: Jan Broz + */ + +#ifndef GRAPH_KRUSKAL +#define GRAPH_KRUSKAL + +#include <core/multipleDispatch.hpp> +#include <graph/GraphClasses.h> + +namespace graph { + +namespace spanningtree { + +class Kruskal : public std::SingleDispatch< graph::Graph, graph::GraphBase > { + + public: + + static Graph kruskal( const Graph & graph ); + + static AdjacencyListUndirectedGraph kruskal( const UndirectedGraph & graph ); + + static Kruskal & getInstance() { + static Kruskal res; + return res; + } + +}; + +} // namespace spanningtree + +} // namespace graph + +#endif // GRAPH_KRUSKAL diff --git a/alib2algo/test-src/graph/TestUtils.cpp b/alib2algo/test-src/graph/TestUtils.cpp new file mode 100644 index 0000000000000000000000000000000000000000..b03e442a58207ad95960c876f9c6e77a47eefe71 --- /dev/null +++ b/alib2algo/test-src/graph/TestUtils.cpp @@ -0,0 +1,40 @@ +#include "TestUtils.h" + +#include <unordered_map> + +void printGraphData( const graph::DirectedGraph & graph ) +{ + unsigned int id; + std::unordered_map<graph::Node, unsigned int> nodes; + std::unordered_map<graph::DirectedEdge, unsigned int> edges; + id = 0; + for (const graph::Node & node : graph.getNodes()) { + nodes[ node ] = id; + id++; + } + id = 0; + for (const graph::DirectedEdge & edge : graph.getEdges()) { + edges[ edge ] = id; + id++; + } + + printf("graph data:\n"); + printf(" node cnt: %lu; edge cnt: %lu\n", graph.nodeCount(), graph.edgeCount()); + printf(" edges:\n"); + for (const graph::DirectedEdge & edge : graph.getEdges()) + printf(" [%u] %u -> %u (%d)\n", edges[edge], nodes[edge.getFromNode()], nodes[edge.getToNode()], graph.getEdgeValue( edge )); + printf(" successors:\n"); + for (const graph::Node & node : graph.getNodes()) { + printf(" [%u] ", nodes[node]); + for (const graph::Node & succ : graph.successors( node )) + printf("%u, ", nodes[succ]); + putchar('\n'); + } + printf(" predecessors:\n"); + for (const graph::Node & node : graph.getNodes()) { + printf(" [%u] ", nodes[node]); + for (const graph::Node & pred : graph.predecessors( node )) + printf("%u, ", nodes[pred]); + putchar('\n'); + } +} diff --git a/alib2algo/test-src/graph/TestUtils.h b/alib2algo/test-src/graph/TestUtils.h new file mode 100644 index 0000000000000000000000000000000000000000..97da47f94dcd6be94e07921571b0ad0c837bdcab --- /dev/null +++ b/alib2algo/test-src/graph/TestUtils.h @@ -0,0 +1,26 @@ +#ifndef TEST_UTILS_H +#define TEST_UTILS_H + +#include <cstdio> +#include <string> +#include <sstream> + +#include <vector> +#include <unordered_map> + +#include <graph/GraphClasses.h> + + +typedef unsigned int uint; + +template< typename Type > +inline std::string toString( const Type & val ) +{ + std::ostringstream oss; + oss << val; + return oss.str(); +} + +void printGraphData( const graph::DirectedGraph & graph ); + +#endif // TEST_UTILS_H diff --git a/alib2algo/test-src/graph/maximumflow/FordFulkersonTest.cpp b/alib2algo/test-src/graph/maximumflow/FordFulkersonTest.cpp new file mode 100644 index 0000000000000000000000000000000000000000..c2487006c357ba8d680d17971df5cdfba23f34a7 --- /dev/null +++ b/alib2algo/test-src/graph/maximumflow/FordFulkersonTest.cpp @@ -0,0 +1,224 @@ +#include "FordFulkersonTest.h" + +#include <graph/maximumflow/FordFulkerson.h> + +#include <exception/AlibException.h> + +#include "../TestUtils.h" + +CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( GraphFordFulkersonTest, "graph" ); +CPPUNIT_TEST_SUITE_REGISTRATION( GraphFordFulkersonTest ); + + +#define NetworkDef( edgeCount ) const struct { uint nodeCnt; uint edgeCnt; uint source; uint sink; struct { uint node1; uint node2; uint capacity; } edges [ edgeCount ]; } +#define FlowDef( edgeCount ) const struct { uint edgeCnt; struct { uint node1; uint node2; int flow; } edges [ edgeCount ]; } + +NetworkDef(1) network1 = { + 2, 1, + 0, 1, + { + { 0, 1, 5 } + } +}; +FlowDef(1) dirflow1 = { + 1, + { + { 0, 1, 5 } + } +}; +FlowDef(1) undirflow1 = { + 1, + { + { 0, 1, 5 } + } +}; + +//---------------------------------- +NetworkDef(4) network2 = { + 4, 4, + 0, 3, + { + { 0, 1, 1 }, + { 0, 2, 4 }, + { 1, 3, 2 }, + { 2, 3, 3 } + } +}; +FlowDef(4) dirflow2 = { + 4, + { + { 0, 1, 1 }, + { 0, 2, 3 }, + { 1, 3, 1 }, + { 2, 3, 3 } + } +}; +FlowDef(4) undirflow2 = { + 4, + { + { 0, 1, 1 }, + { 0, 2, 3 }, + { 1, 3, 1 }, + { 2, 3, 3 } + } +}; + +//---------------------------------- +NetworkDef(5) network3 = { + 4, 5, + 0, 3, + { + { 0, 1, 5 }, + { 0, 2, 6 }, + { 1, 2, 2 }, + { 1, 3, 7 }, + { 2, 3, 4 } + } +}; +FlowDef(5) dirflow3 = { + 5, + { + { 0, 1, 5 }, + { 0, 2, 4 }, + { 1, 2, 0 }, + { 1, 3, 5 }, + { 2, 3, 4 } + } +}; +FlowDef(5) undirflow3 = { + 5, + { + { 0, 1, 5 }, + { 0, 2, 6 }, + { 1, 2, -2 }, + { 1, 3, 7 }, + { 2, 3, 4 } + } +}; + +//---------------------------------- +NetworkDef(10) network4 = { + 7, 10, + 0, 6, + { + { 0, 3, 12 }, + { 0, 1, 6 }, + { 1, 3, 2 }, + { 1, 2, 8 }, + { 2, 4, 7 }, + { 3, 4, 1 }, + { 3, 5, 4 }, + { 3, 6, 6 }, + { 4, 6, 7 }, + { 5, 6, 3 } + } +}; +FlowDef(10) dirflow4 = { + 10, + { + { 0, 3, 10 }, + { 0, 1, 6 }, + { 1, 3, 0 }, + { 1, 2, 6 }, + { 2, 4, 6 }, + { 3, 4, 1 }, + { 3, 5, 3 }, + { 3, 6, 6 }, + { 4, 6, 7 }, + { 5, 6, 3 } + } +}; +FlowDef(10) undirflow4 = { + 10, + { + { 0, 3, 10 }, + { 0, 1, 6 }, + { 1, 3, 0 }, + { 1, 2, 6 }, + { 2, 4, 6 }, + { 3, 4, 1 }, + { 3, 5, 3 }, + { 3, 6, 6 }, + { 4, 6, 7 }, + { 5, 6, 3 } + } +}; + + +template< typename Type1, typename Type2 > +void testDirNetwork( uint netID, const Type1 * netDef, const Type2 * flowDef ) +{ + graph::AdjacencyListDirectedGraph graph; + std::vector<graph::Node> id2node( netDef->nodeCnt ); + graph::Node source, sink; + uint i, j; + + for (i = 0; i < netDef->nodeCnt; i++) { + graph::Node node( std::string("n")+toString(i) ); + graph.addNode( node ); + id2node[ i ] = node; + if (i == netDef->source) + source = node; + if (i == netDef->sink) + sink = node; + } + for (i = 0; i < netDef->edgeCnt; i++) { + graph::DirectedEdge edge( id2node[ netDef->edges[i].node1 ], id2node[ netDef->edges[i].node2 ], std::string("e")+toString(i) ); + graph.addEdge( edge, netDef->edges[i].capacity ); + } + + std::cout << "testing network #" << netID << std::endl; + + graph::maximumflow::Flow flow = graph::maximumflow::FordFulkerson::fordfulkerson( graph, source, sink ); + + for (uint id = 0; id < flowDef->edgeCnt; id++) { + i = flowDef->edges[id].node1; + j = flowDef->edges[id].node2; + CPPUNIT_ASSERT_EQUAL( flowDef->edges[id].flow, flow[ id2node[i] ][ id2node[j] ] ); + } +} + +template< typename Type1, typename Type2 > +void testUndirNetwork( uint netID, const Type1 * netDef, const Type2 * flowDef ) +{ + graph::AdjacencyListUndirectedGraph graph; + std::vector<graph::Node> id2node( netDef->nodeCnt ); + graph::Node source, sink; + uint i, j; + + for (i = 0; i < netDef->nodeCnt; i++) { + graph::Node node( std::string("n")+toString(i) ); + graph.addNode( node ); + id2node[ i ] = node; + if (i == netDef->source) + source = node; + if (i == netDef->sink) + sink = node; + } + for (i = 0; i < netDef->edgeCnt; i++) { + graph::UndirectedEdge edge( id2node[ netDef->edges[i].node1 ], id2node[ netDef->edges[i].node2 ], std::string("e")+toString(i) ); + graph.addEdge( edge, netDef->edges[i].capacity ); + } + + std::cout << "testing network #" << netID << std::endl; + + graph::maximumflow::Flow flow = graph::maximumflow::FordFulkerson::fordfulkerson( graph, source, sink ); + + for (uint id = 0; id < flowDef->edgeCnt; id++) { + i = flowDef->edges[id].node1; + j = flowDef->edges[id].node2; + CPPUNIT_ASSERT_EQUAL( flowDef->edges[id].flow, flow[ id2node[i] ][ id2node[j] ] ); + } +} + +void GraphFordFulkersonTest::runTests() +{ + testDirNetwork ( 1, &network1, &dirflow1 ); + testUndirNetwork( 2, &network1, &undirflow1 ); + testDirNetwork ( 3, &network2, &dirflow2 ); + testUndirNetwork( 4, &network2, &undirflow2 ); + testDirNetwork ( 5, &network3, &dirflow3 ); + testUndirNetwork( 6, &network3, &undirflow3 ); + testDirNetwork ( 7, &network4, &dirflow4 ); + testUndirNetwork( 8, &network4, &undirflow4 ); +} diff --git a/alib2algo/test-src/graph/maximumflow/FordFulkersonTest.h b/alib2algo/test-src/graph/maximumflow/FordFulkersonTest.h new file mode 100644 index 0000000000000000000000000000000000000000..38ce10f6b39dc95b3e8940ba437a244c1c2612e3 --- /dev/null +++ b/alib2algo/test-src/graph/maximumflow/FordFulkersonTest.h @@ -0,0 +1,20 @@ +#ifndef FORD_FULKERSON_TEST_H +#define FORD_FULKERSON_TEST_H + +#include <cppunit/extensions/HelperMacros.h> + +class GraphFordFulkersonTest : public CppUnit::TestFixture { + + CPPUNIT_TEST_SUITE( GraphFordFulkersonTest ); + + CPPUNIT_TEST( runTests ); + + CPPUNIT_TEST_SUITE_END(); + + public: + + void runTests(); + +}; + +#endif // FORD_FULKERSON_TEST_H diff --git a/alib2algo/test-src/graph/minimumcut/FordFulkersonTest.cpp b/alib2algo/test-src/graph/minimumcut/FordFulkersonTest.cpp new file mode 100644 index 0000000000000000000000000000000000000000..6ea36f9418301765b8bfda765764cf11e2a054a5 --- /dev/null +++ b/alib2algo/test-src/graph/minimumcut/FordFulkersonTest.cpp @@ -0,0 +1,181 @@ +#include "FordFulkersonTest.h" + +#include <graph/minimumcut/FordFulkerson.h> + +#include <utility> + +#include <exception/AlibException.h> + +#include "../TestUtils.h" + +CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( GraphFordFulkersonCutTest, "graph" ); +CPPUNIT_TEST_SUITE_REGISTRATION( GraphFordFulkersonCutTest ); + + +#define NetworkDef( edgeCount ) const struct { uint nodeCnt; uint edgeCnt; uint source; uint sink; struct { uint node1; uint node2; uint capacity; } edges [ edgeCount ]; } +struct NodePair { + uint node1ID; + uint node2ID; +}; + +NetworkDef(1) network1 = { + 2, 1, + 0, 1, + { + { 0, 1, 5 } + } +}; +NodePair dircut1 [1] = { + { 0, 1 } +}; +NodePair undircut1 [1] = { + { 0, 1 } +}; + +//---------------------------------- +NetworkDef(4) network2 = { + 4, 4, + 0, 3, + { + { 0, 1, 1 }, + { 0, 2, 4 }, + { 1, 3, 2 }, + { 2, 3, 3 } + } +}; +NodePair dircut2 [2] = { + { 0, 1 }, + { 2, 3 } +}; +NodePair undircut2 [2] = { + { 0, 1 }, + { 2, 3 } +}; + +//---------------------------------- +NetworkDef(5) network3 = { + 4, 5, + 0, 3, + { + { 0, 1, 5 }, + { 0, 2, 6 }, + { 1, 3, 7 }, + { 2, 1, 2 }, + { 2, 3, 4 } + } +}; +NodePair dircut3 [2] = { + { 0, 1 }, + { 0, 2 } +}; +NodePair undircut3 [2] = { + { 0, 1 }, + { 0, 2 } +}; + +//---------------------------------- +NetworkDef(10) network4 = { + 7, 10, + 0, 6, + { + { 0, 3, 12 }, + { 0, 1, 6 }, + { 1, 3, 2 }, + { 1, 2, 8 }, + { 2, 4, 7 }, + { 3, 4, 1 }, + { 3, 5, 4 }, + { 3, 6, 6 }, + { 4, 6, 7 }, + { 5, 6, 3 } + } +}; +NodePair dircut4 [5] = { + { 0, 1 }, + { 1, 3 }, + { 3, 4 }, + { 3, 6 }, + { 5, 6 } +}; +NodePair undircut4 [3] = { + { 5, 6 }, + { 3, 6 }, + { 4, 6 } +}; + + +template< typename Type1, typename Type2 > +void testDirNetwork( uint netID, const Type1 * netDef, const Type2 cutDef, uint cutSize ) +{ + graph::AdjacencyListDirectedGraph graph; + std::vector<graph::Node> id2node( netDef->nodeCnt ); + graph::Node source, sink; + uint i; + + for (i = 0; i < netDef->nodeCnt; i++) { + graph::Node node( std::string("n")+toString(i) ); + graph.addNode( node ); + id2node[ i ] = node; + if (i == netDef->source) + source = node; + if (i == netDef->sink) + sink = node; + } + for (i = 0; i < netDef->edgeCnt; i++) { + graph::DirectedEdge edge( id2node[ netDef->edges[i].node1 ], id2node[ netDef->edges[i].node2 ], std::string("e")+toString(i) ); + graph.addEdge( edge, netDef->edges[i].capacity ); + } + + std::cout << "testing network #" << netID << std::endl; + + graph::minimumcut::Cut cut = graph::minimumcut::FordFulkerson::fordfulkerson( graph, source, sink ); + + CPPUNIT_ASSERT_EQUAL( cutSize, (uint)cut.size() ); + for (uint i = 0; i < cutSize; i++) { + CPPUNIT_ASSERT( cut.find( std::make_pair( id2node[cutDef[i].node1ID], id2node[cutDef[i].node2ID] ) ) != cut.end() ); + } +} + +template< typename Type1, typename Type2 > +void testUndirNetwork( uint netID, const Type1 * netDef, const Type2 cutDef, uint cutSize ) +{ + graph::AdjacencyListUndirectedGraph graph; + std::vector<graph::Node> id2node( netDef->nodeCnt ); + graph::Node source, sink; + uint i; + + for (i = 0; i < netDef->nodeCnt; i++) { + graph::Node node( std::string("n")+toString(i) ); + graph.addNode( node ); + id2node[ i ] = node; + if (i == netDef->source) + source = node; + if (i == netDef->sink) + sink = node; + } + for (i = 0; i < netDef->edgeCnt; i++) { + graph::UndirectedEdge edge( id2node[ netDef->edges[i].node1 ], id2node[ netDef->edges[i].node2 ], std::string("e")+toString(i) ); + graph.addEdge( edge, netDef->edges[i].capacity ); + } + + std::cout << "testing network #" << netID << std::endl; + + graph::minimumcut::Cut cut = graph::minimumcut::FordFulkerson::fordfulkerson( graph, source, sink ); + + CPPUNIT_ASSERT_EQUAL( cutSize, (uint)cut.size() ); + for (uint i = 0; i < cutSize; i++) { + CPPUNIT_ASSERT( cut.find( std::make_pair( id2node[cutDef[i].node1ID], id2node[cutDef[i].node2ID] ) ) != cut.end() ); + } +} + +void GraphFordFulkersonCutTest::runTests() +{ + testDirNetwork ( 1, &network1, dircut1, sizeof(dircut1)/sizeof(dircut1[0]) ); + testUndirNetwork( 2, &network1, undircut1, sizeof(undircut1)/sizeof(undircut1[0]) ); + testDirNetwork ( 3, &network2, dircut2, sizeof(dircut2)/sizeof(dircut2[0]) ); + testUndirNetwork( 4, &network2, undircut2, sizeof(undircut2)/sizeof(undircut2[0]) ); + testDirNetwork ( 5, &network3, dircut3, sizeof(dircut3)/sizeof(dircut3[0]) ); + testUndirNetwork( 6, &network3, undircut3, sizeof(undircut3)/sizeof(undircut3[0]) ); + testDirNetwork ( 7, &network4, dircut4, sizeof(dircut4)/sizeof(dircut4[0]) ); + testUndirNetwork( 8, &network4, undircut4, sizeof(undircut4)/sizeof(undircut4[0]) ); +} diff --git a/alib2algo/test-src/graph/minimumcut/FordFulkersonTest.h b/alib2algo/test-src/graph/minimumcut/FordFulkersonTest.h new file mode 100644 index 0000000000000000000000000000000000000000..08e392d4265f7f88fa692b800e135f440ff87410 --- /dev/null +++ b/alib2algo/test-src/graph/minimumcut/FordFulkersonTest.h @@ -0,0 +1,20 @@ +#ifndef FORD_FULKERSON_TEST_CUT_H +#define FORD_FULKERSON_TEST_CUT_H + +#include <cppunit/extensions/HelperMacros.h> + +class GraphFordFulkersonCutTest : public CppUnit::TestFixture { + + CPPUNIT_TEST_SUITE( GraphFordFulkersonCutTest ); + + CPPUNIT_TEST( runTests ); + + CPPUNIT_TEST_SUITE_END(); + + public: + + void runTests(); + +}; + +#endif // FORD_FULKERSON_TEST_CUT_H diff --git a/alib2algo/test-src/graph/spanningtree/EdmondsTest.cpp b/alib2algo/test-src/graph/spanningtree/EdmondsTest.cpp new file mode 100644 index 0000000000000000000000000000000000000000..c4d14579f912dd2c5605e9a64a6e225cc515608e --- /dev/null +++ b/alib2algo/test-src/graph/spanningtree/EdmondsTest.cpp @@ -0,0 +1,187 @@ +#include "EdmondsTest.h" + +#include <graph/spanningtree/Edmonds.h> + +#include <graph/generate/RandomGraphFactory.h> +#include <exception/AlibException.h> +#include "../TestUtils.h" + +CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( GraphEdmondsTest, "graph" ); +CPPUNIT_TEST_SUITE_REGISTRATION( GraphEdmondsTest ); + + +#define GraphDef( edgeCount ) const struct { uint nodeCnt; uint edgeCnt; struct { uint node1; uint node2; uint weight; } edges [ edgeCount ]; } + +GraphDef(1) graph1 = { + 2, 1, + { + { 0, 1, 5 } + } +}; +GraphDef(1) spanningTree1 = { + 2, 1, + { + { 0, 1, 5 } + } +}; + +//---------------------------------- +GraphDef(2) graph2 = { + 3, 2, + { + { 0, 1, 5 }, + { 1, 2, 6 } + } +}; +GraphDef(2) spanningTree2 = { + 3, 2, + { + { 0, 1, 5 }, + { 1, 2, 6 } + } +}; + +//---------------------------------- +GraphDef(2) graph3 = { + 3, 2, + { + { 0, 1, 5 }, + { 0, 2, 6 } + } +}; +GraphDef(2) spanningTree3 = { + 3, 2, + { + { 0, 1, 5 }, + { 0, 2, 6 } + } +}; + +//---------------------------------- +GraphDef(3) graph4 = { + 4, 3, + { + { 0, 1, 5 }, + { 0, 3, 6 }, + { 2, 0, 7 } + } +}; +GraphDef(3) spanningTree4 = { + 4, 3, + { + { 0, 1, 5 }, + { 0, 3, 6 }, + { 2, 0, 7 } + } +}; + +//---------------------------------- +GraphDef(3) graph5 = { + 3, 3, + { + { 0, 1, 7 }, + { 1, 2, 6 }, + { 1, 2, 5 } + } +}; +GraphDef(2) spanningTree5 = { + 3, 2, + { + { 0, 1, 7 }, + { 1, 2, 6 }, + } +}; + +//---------------------------------- +GraphDef(5) graph6 = { + 4, 5, + { + { 0, 1, 4 }, + { 0, 2, 1 }, + { 0, 3, 2 }, + { 1, 3, 3 }, + { 3, 2, 5 } + } +}; +GraphDef(3) spanningTree6 = { + 4, 3, + { + { 0, 2, 1 }, + { 0, 1, 4 }, + { 0, 3, 2 } + } +}; + +//---------------------------------- +GraphDef(10) graph7 = { + 6, 10, + { + { 0, 1, 5 }, + { 0, 2, 2 }, + { 0, 4, 1 }, + { 1, 3, 4 }, + { 1, 5, 3 }, + { 2, 3, 2 }, + { 3, 4, 11 }, + { 4, 2, 1 }, + { 4, 1, 8 }, + { 5, 0, 9 } + } +}; +GraphDef(5) spanningTree7 = { + 6, 5, + { + { 0, 4, 1 }, + { 4, 2, 1 }, + { 2, 3, 2 }, + { 0, 1, 5 }, + { 1, 5, 3 } + } +}; + + +template< typename Type1, typename Type2 > +void testGraph( uint graphID, const Type1 * graphDef, const Type2 * spanTreeDef ) +{ + std::vector<graph::Node> id2node( graphDef->nodeCnt ); + std::vector<graph::DirectedEdge> id2edge( graphDef->edgeCnt ); + std::unordered_map<graph::Node,uint> node2id; + std::unordered_map<graph::DirectedEdge,uint> edge2id; + + graph::AdjacencyListDirectedGraph graph; + + for (uint i = 0; i < graphDef->nodeCnt; i++) { + graph::Node node( std::string("n")+toString(i) ); + graph.addNode( node ); + id2node[ i ] = node; + node2id[ node ] = i; + } + for (uint i = 0; i < graphDef->edgeCnt; i++) { + graph::DirectedEdge edge( id2node[ graphDef->edges[i].node1 ], id2node[ graphDef->edges[i].node2 ], std::string("e")+toString(i) ); + graph.addEdge( edge, graphDef->edges[i].weight ); + id2edge[ i ] = edge; + edge2id[ edge ] = i; + } + + std::cout << "testing spanning tree #" << graphID << std::endl; + + graph::AdjacencyListDirectedGraph spanningTree = graph::spanningtree::Edmonds::edmonds( graph ); + + CPPUNIT_ASSERT_EQUAL_MESSAGE( "invalid node count", spanTreeDef->nodeCnt, (uint)spanningTree.nodeCount() ); + CPPUNIT_ASSERT_EQUAL_MESSAGE( "invalid edge count", spanTreeDef->edgeCnt, (uint)spanningTree.edgeCount() ); + for (uint i = 0; i < spanTreeDef->edgeCnt; i++) { + CPPUNIT_ASSERT_EQUAL_MESSAGE( "missing edge: "+toString(spanTreeDef->edges[i].node1)+","+toString(spanTreeDef->edges[i].node2), + true, spanningTree.hasEdge( id2node[ spanTreeDef->edges[i].node1 ], id2node[ spanTreeDef->edges[i].node2 ] ) ); + } +} + +void GraphEdmondsTest::runTests() +{ + testGraph( 1, &graph1, &spanningTree1 ); + testGraph( 2, &graph2, &spanningTree2 ); + testGraph( 3, &graph3, &spanningTree3 ); + testGraph( 4, &graph4, &spanningTree4 ); + testGraph( 5, &graph5, &spanningTree5 ); + testGraph( 6, &graph6, &spanningTree6 ); + testGraph( 7, &graph7, &spanningTree7 ); +} diff --git a/alib2algo/test-src/graph/spanningtree/EdmondsTest.h b/alib2algo/test-src/graph/spanningtree/EdmondsTest.h new file mode 100644 index 0000000000000000000000000000000000000000..0ea95c1c7f906678596bfb96208672aea4b6b677 --- /dev/null +++ b/alib2algo/test-src/graph/spanningtree/EdmondsTest.h @@ -0,0 +1,20 @@ +#ifndef EDMONDS_TEST_H +#define EDMONDS_TEST_H + +#include <cppunit/extensions/HelperMacros.h> + +class GraphEdmondsTest : public CppUnit::TestFixture { + + CPPUNIT_TEST_SUITE( GraphEdmondsTest ); + + CPPUNIT_TEST( runTests ); + + CPPUNIT_TEST_SUITE_END(); + + public: + + void runTests(); + +}; + +#endif // EDMONDS_TEST_H diff --git a/alib2algo/test-src/graph/spanningtree/KruskalTest.cpp b/alib2algo/test-src/graph/spanningtree/KruskalTest.cpp new file mode 100644 index 0000000000000000000000000000000000000000..134becbcc4de1fcd8819c345717189812b0d4bc6 --- /dev/null +++ b/alib2algo/test-src/graph/spanningtree/KruskalTest.cpp @@ -0,0 +1,160 @@ +#include "KruskalTest.h" + +#include <graph/spanningtree/Kruskal.h> + +#include <graph/generate/RandomGraphFactory.h> +#include <exception/AlibException.h> +#include "../TestUtils.h" + +CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( GraphKruskalTest, "graph" ); +CPPUNIT_TEST_SUITE_REGISTRATION( GraphKruskalTest ); + +#define GraphDef( edgeCount ) const struct { uint nodeCnt; uint edgeCnt; struct { uint node1; uint node2; uint weight; } edges [ edgeCount ]; } + +GraphDef(1) graph1 = { + 2, 1, + { + { 0, 1, 5 } + } +}; +GraphDef(1) spanningTree1 = { + 2, 1, + { + { 0, 1, 5 } + } +}; + +//---------------------------------- +GraphDef(3) graph2 = { + 3, 3, + { + { 0, 1, 7 }, + { 1, 2, 6 }, + { 1, 2, 5 } + } +}; +GraphDef(2) spanningTree2 = { + 3, 2, + { + { 0, 1, 7 }, + { 1, 2, 6 }, + } +}; + +//---------------------------------- +GraphDef(5) graph3 = { + 4, 5, + { + { 0, 1, 4 }, + { 0, 2, 1 }, + { 0, 3, 2 }, + { 1, 3, 3 }, + { 3, 2, 5 } + } +}; +GraphDef(3) spanningTree3 = { + 4, 3, + { + { 0, 2, 1 }, + { 0, 3, 2 }, + { 1, 3, 3 } + } +}; + +GraphDef(24) graph4 = { + 12, 24, + { + { 0, 1, 1 }, + { 1, 2, 2 }, + { 2, 3, 2 }, + { 0, 4, 2 }, + { 1, 4, 4 }, + { 1, 5, 3 }, + { 2, 5, 1 }, + { 2, 6, 3 }, + { 3, 6, 4 }, + { 3, 7, 5 }, + { 4, 5, 1 }, + { 5, 6, 5 }, + { 6, 7, 2 }, + { 4, 8, 5 }, + { 4, 9, 2 }, + { 5, 9, 3 }, + { 5,10, 4 }, + { 6,10, 1 }, + { 6,11, 2 }, + { 7,11, 5 }, + { 8, 9, 6 }, + { 9,10, 2 }, + {10,11, 4 } + } +}; +GraphDef(11) spanningTree4 = { + 12, 11, + { + { 0, 1, 1 }, + { 1, 2, 2 }, + { 2, 3, 2 }, + { 2, 5, 1 }, + { 4, 5, 1 }, + { 6, 7, 2 }, + { 4, 8, 5 }, + { 4, 9, 2 }, + { 6,10, 1 }, + { 6,11, 3 }, + { 9,10, 1 } + } +}; + + + +template< typename Type1, typename Type2 > +void testGraph( uint graphID, const Type1 * graphDef, const Type2 * spanTreeDef ) +{ + std::vector<graph::Node> id2node( graphDef->nodeCnt ); + std::vector<graph::UndirectedEdge> id2edge( graphDef->edgeCnt ); + std::unordered_map<graph::Node,uint> node2id; + std::unordered_map<graph::UndirectedEdge,uint> edge2id; + + graph::AdjacencyListUndirectedGraph graph; + + for (uint i = 0; i < graphDef->nodeCnt; i++) { + graph::Node node( std::string("n")+toString(i) ); + graph.addNode( node ); + id2node[ i ] = node; + node2id[ node ] = i; + } + for (uint i = 0; i < graphDef->edgeCnt; i++) { + graph::UndirectedEdge edge( id2node[ graphDef->edges[i].node1 ], id2node[ graphDef->edges[i].node2 ], std::string("e")+toString(i) ); + graph.addEdge( edge, graphDef->edges[i].weight ); + id2edge[ i ] = edge; + edge2id[ edge ] = i; + } + + std::cout << "testing spanning tree #" << graphID << std::endl; + + graph::AdjacencyListUndirectedGraph spanningTree = graph::spanningtree::Kruskal::kruskal( graph ); + + CPPUNIT_ASSERT_EQUAL_MESSAGE( "invalid node count", spanTreeDef->nodeCnt, (uint)spanningTree.nodeCount() ); + CPPUNIT_ASSERT_EQUAL_MESSAGE( "invalid edge count", spanTreeDef->edgeCnt, (uint)spanningTree.edgeCount() ); + for (uint i = 0; i < spanTreeDef->edgeCnt; i++) { + CPPUNIT_ASSERT_EQUAL_MESSAGE( "missing edge: "+toString(spanTreeDef->edges[i].node1)+","+toString(spanTreeDef->edges[i].node2), + true, spanningTree.hasEdge( id2node[ spanTreeDef->edges[i].node1 ], id2node[ spanTreeDef->edges[i].node2 ] ) ); + } +} + +void GraphKruskalTest::runTests() +{ + testGraph( 1, &graph1, &spanningTree1 ); + testGraph( 2, &graph2, &spanningTree2 ); + testGraph( 3, &graph3, &spanningTree3 ); + testGraph( 4, &graph4, &spanningTree4 ); + + std::cout << "testing random big graphs" << std::endl; + int nodeCount = 8; + for (int i = 0; i < 8; i++) { + graph::AdjacencyListUndirectedGraph graph = graph::generate::RandomGraphFactory::generateUndirectedGraph( nodeCount, nodeCount*2 ); + graph::spanningtree::Kruskal::kruskal( graph ); + nodeCount *= 2; + } +} diff --git a/alib2algo/test-src/graph/spanningtree/KruskalTest.h b/alib2algo/test-src/graph/spanningtree/KruskalTest.h new file mode 100644 index 0000000000000000000000000000000000000000..e84209cbe7ef931e51ca1ab93c188e441506c427 --- /dev/null +++ b/alib2algo/test-src/graph/spanningtree/KruskalTest.h @@ -0,0 +1,20 @@ +#ifndef KRUSKAL_TEST_H +#define KRUSKAL_TEST_H + +#include <cppunit/extensions/HelperMacros.h> + +class GraphKruskalTest : public CppUnit::TestFixture { + + CPPUNIT_TEST_SUITE( GraphKruskalTest ); + + CPPUNIT_TEST( runTests ); + + CPPUNIT_TEST_SUITE_END(); + + public: + + void runTests(); + +}; + +#endif // KRUSKAL_TEST_H