From 78cf2258a922e074f830aa23d2e1cd582da1b4bb Mon Sep 17 00:00:00 2001 From: David Rosca <roscadav@fit.cvut.cz> Date: Sun, 8 Mar 2015 15:13:55 +0100 Subject: [PATCH] Graph shortestpath algo: Add Dijkstra algorithm + tests --- alib2algo/src/graph/shortestpath/Dijkstra.cpp | 155 ++++++++++++++ alib2algo/src/graph/shortestpath/Dijkstra.h | 39 ++++ .../graph/shortestpath/DijkstraTest.cpp | 191 ++++++++++++++++++ .../graph/shortestpath/DijkstraTest.h | 22 ++ 4 files changed, 407 insertions(+) create mode 100644 alib2algo/src/graph/shortestpath/Dijkstra.cpp create mode 100644 alib2algo/src/graph/shortestpath/Dijkstra.h create mode 100644 alib2algo/test-src/graph/shortestpath/DijkstraTest.cpp create mode 100644 alib2algo/test-src/graph/shortestpath/DijkstraTest.h diff --git a/alib2algo/src/graph/shortestpath/Dijkstra.cpp b/alib2algo/src/graph/shortestpath/Dijkstra.cpp new file mode 100644 index 0000000000..098b5abfd7 --- /dev/null +++ b/alib2algo/src/graph/shortestpath/Dijkstra.cpp @@ -0,0 +1,155 @@ +#include "Dijkstra.h" + +#include <exception/AlibException.h> + +namespace graph +{ + +namespace shortestpath +{ + +class Data +{ +public: + Node start; + Dijkstra::Result out; +}; + +class Value +{ +public: + Value(const Node &node, int value) + : node(node) + , value(value) + { + } + + bool operator<(const Value &other) const + { + return value < other.value; + } + + Node node; + int value; +}; + +typedef std::unordered_map<Node, std::unordered_map<Node, int>> weights_t; + +static weights_t weights(const DirectedGraph &g) +{ + int val; + weights_t w; + + for (const DirectedEdge &e : g.getEdges()) { + auto &map = w[e.getFromNode()]; + auto search = map.find(e.getToNode()); + + if (search == map.end()) { + val = g.getEdgeValue(e); + map[e.getToNode()] = val; + } else { + val = std::min(search->second, g.getEdgeValue(e)); + search->second = val; + } + + if (val < 0) { + throw exception::AlibException("Dijkstra: Found negative value!"); + } + } + + return w; +} + +static weights_t weights(const UndirectedGraph &g) +{ + int val; + weights_t w; + + for (const UndirectedEdge &e : g.getEdges()) { + auto &map = w[e.getFirstNode()]; + auto search = map.find(e.getSecondNode()); + + if (search == map.end()) { + val = g.getEdgeValue(e); + map[e.getSecondNode()] = val; + w[e.getSecondNode()][e.getFirstNode()] = val; + } else { + val = std::min(search->second, g.getEdgeValue(e)); + search->second = val; + w[e.getSecondNode()][e.getFirstNode()] = val; + } + + if (val < 0) { + throw exception::AlibException("Dijkstra: Found negative value!"); + } + } + + return w; +} + +template <typename T> +static Dijkstra::Result dijkstra_impl(const T &graph, const Node &start) +{ + std::set<Value> q; // priority queue + Dijkstra::Result d; // distances + weights_t w = weights(graph); // minimum weights + + d[start] = 0; + q.insert(Value(start, 0)); + + while (!q.empty()) { + Node u = q.begin()->node; q.erase(q.begin()); + + for (const Node &v : graph.neighbors(u)) { + int val = d.at(u) + w[u][v]; + auto search = d.find(v); + + if (search == d.end() || search->second > val) { + if (search != d.end()) { + q.erase(q.find(Value(v, search->second))); + } + + q.insert(Value(v, val)); + d[v] = val; + } + } + } + + return d; +} + +Dijkstra::Result Dijkstra::dijkstra(const Graph &graph, const Node &start) +{ + Data data; + data.start = start; + graph.getData().Accept(static_cast<void*>(&data), DIJKSTRA); + return data.out; +} + +Dijkstra::Result Dijkstra::dijkstra(const DirectedGraph &graph, const Node &start) +{ + return dijkstra_impl(graph, start); +} + +Dijkstra::Result Dijkstra::dijkstra(const UndirectedGraph &graph, const Node &start) +{ + return dijkstra_impl(graph, start); +} + +void Dijkstra::Visit(void *data, const DirectedGraph &graph) const +{ + Data d = *static_cast<Data*>(data); + d.out = dijkstra(graph, d.start); +} + +void Dijkstra::Visit(void *data, const UndirectedGraph &graph) const +{ + Data d = *static_cast<Data*>(data); + d.out = dijkstra(graph, d.start); +} + +const Dijkstra Dijkstra::DIJKSTRA; + +} // namespace shortestpath + +} // namespace graph diff --git a/alib2algo/src/graph/shortestpath/Dijkstra.h b/alib2algo/src/graph/shortestpath/Dijkstra.h new file mode 100644 index 0000000000..d9373e3ca3 --- /dev/null +++ b/alib2algo/src/graph/shortestpath/Dijkstra.h @@ -0,0 +1,39 @@ +#ifndef GRAPH_DIJKSTRA_H_ +#define GRAPH_DIJKSTRA_H_ + +#include <unordered_map> + +#include <graph/Graph.h> +#include <graph/directed/DirectedGraph.h> +#include <graph/undirected/UndirectedGraph.h> + +namespace graph +{ + +namespace shortestpath +{ + +// Dijkstra only works on graphs without negative-weight edges (>= 0) + +class Dijkstra : public graph::VisitableGraphBase::const_visitor_type +{ +public: + typedef std::unordered_map<Node, int> Result; + + static Result dijkstra(const Graph &graph, const Node &start); + + static Result dijkstra(const DirectedGraph &graph, const Node &start); + static Result dijkstra(const UndirectedGraph &graph, const Node &start); + +private: + void Visit(void *data, const DirectedGraph &graph) const; + void Visit(void *data, const UndirectedGraph &graph) const; + + static const Dijkstra DIJKSTRA; +}; + +} // namespace shortestpath + +} // namespace graph + +#endif // GRAPH_DIJKSTRA_H_ diff --git a/alib2algo/test-src/graph/shortestpath/DijkstraTest.cpp b/alib2algo/test-src/graph/shortestpath/DijkstraTest.cpp new file mode 100644 index 0000000000..6b1b2b65bf --- /dev/null +++ b/alib2algo/test-src/graph/shortestpath/DijkstraTest.cpp @@ -0,0 +1,191 @@ +#include "DijkstraTest.h" + +#include "graph/shortestpath/Dijkstra.h" + +#include <exception/AlibException.h> + +CPPUNIT_TEST_SUITE_REGISTRATION(GraphDijkstraTest); + +void GraphDijkstraTest::testSimple() +{ + // Common + graph::shortestpath::Dijkstra::Result res; + graph::Node n1("n1"); + graph::Node n2("n2"); + graph::Node n3("n3"); + graph::Node n4("n4"); + graph::Node n5("n5"); + graph::Node n6("n6"); + + // Directed + graph::DirectedGraph dg; + dg.addEdge(graph::DirectedEdge(n1, n2), 2); + dg.addEdge(graph::DirectedEdge(n1, n3), 2); + dg.addEdge(graph::DirectedEdge(n1, n4), 1); + dg.addEdge(graph::DirectedEdge(n1, n5), 6); + dg.addEdge(graph::DirectedEdge(n2, n5), 3); + dg.addEdge(graph::DirectedEdge(n5, n6), 2); + + res = graph::shortestpath::Dijkstra::dijkstra(dg, n1); + + CPPUNIT_ASSERT_EQUAL(0, res[n1]); + CPPUNIT_ASSERT_EQUAL(2, res[n2]); + CPPUNIT_ASSERT_EQUAL(2, res[n3]); + CPPUNIT_ASSERT_EQUAL(1, res[n4]); + CPPUNIT_ASSERT_EQUAL(5, res[n5]); + CPPUNIT_ASSERT_EQUAL(7, res[n6]); + + // Undirected + graph::UndirectedGraph ug; + ug.addEdge(graph::UndirectedEdge(n1, n2), 2); + ug.addEdge(graph::UndirectedEdge(n1, n3), 2); + ug.addEdge(graph::UndirectedEdge(n1, n4), 1); + ug.addEdge(graph::UndirectedEdge(n1, n5), 6); + ug.addEdge(graph::UndirectedEdge(n2, n5), 3); + ug.addEdge(graph::UndirectedEdge(n5, n6), 2); + + res = graph::shortestpath::Dijkstra::dijkstra(ug, n1); + + CPPUNIT_ASSERT_EQUAL(0, res[n1]); + CPPUNIT_ASSERT_EQUAL(2, res[n2]); + CPPUNIT_ASSERT_EQUAL(2, res[n3]); + CPPUNIT_ASSERT_EQUAL(1, res[n4]); + CPPUNIT_ASSERT_EQUAL(5, res[n5]); + CPPUNIT_ASSERT_EQUAL(7, res[n6]); +} + +void GraphDijkstraTest::testCycle() +{ + // Common + graph::shortestpath::Dijkstra::Result res; + graph::Node n1("n1"); + graph::Node n2("n2"); + graph::Node n3("n3"); + graph::Node n4("n4"); + graph::Node n5("n5"); + graph::Node n6("n6"); + + // Directed + graph::DirectedGraph dg; + dg.addEdge(graph::DirectedEdge(n1, n2), 2); + dg.addEdge(graph::DirectedEdge(n1, n3), 2); + dg.addEdge(graph::DirectedEdge(n1, n4), 1); + dg.addEdge(graph::DirectedEdge(n1, n5), 6); + dg.addEdge(graph::DirectedEdge(n2, n5), 3); + dg.addEdge(graph::DirectedEdge(n5, n6), 2); + dg.addEdge(graph::DirectedEdge(n6, n1), 1); + + res = graph::shortestpath::Dijkstra::dijkstra(dg, n1); + + CPPUNIT_ASSERT_EQUAL(0, res[n1]); + CPPUNIT_ASSERT_EQUAL(2, res[n2]); + CPPUNIT_ASSERT_EQUAL(2, res[n3]); + CPPUNIT_ASSERT_EQUAL(1, res[n4]); + CPPUNIT_ASSERT_EQUAL(5, res[n5]); + CPPUNIT_ASSERT_EQUAL(7, res[n6]); + + // Undirected + graph::UndirectedGraph ug; + ug.addEdge(graph::UndirectedEdge(n1, n2), 2); + ug.addEdge(graph::UndirectedEdge(n1, n3), 2); + ug.addEdge(graph::UndirectedEdge(n1, n4), 1); + ug.addEdge(graph::UndirectedEdge(n1, n5), 6); + ug.addEdge(graph::UndirectedEdge(n2, n5), 3); + ug.addEdge(graph::UndirectedEdge(n5, n6), 2); + ug.addEdge(graph::UndirectedEdge(n6, n1), 1); + + res = graph::shortestpath::Dijkstra::dijkstra(ug, n1); + + CPPUNIT_ASSERT_EQUAL(0, res[n1]); + CPPUNIT_ASSERT_EQUAL(2, res[n2]); + CPPUNIT_ASSERT_EQUAL(2, res[n3]); + CPPUNIT_ASSERT_EQUAL(1, res[n4]); + CPPUNIT_ASSERT_EQUAL(5, res[n5]); + CPPUNIT_ASSERT_EQUAL(1, res[n6]); +} + +void GraphDijkstraTest::testMultiEdge() +{ + // Common + graph::shortestpath::Dijkstra::Result res; + graph::Node n1("n1"); + graph::Node n2("n2"); + graph::Node n3("n3"); + graph::Node n4("n4"); + graph::Node n5("n5"); + graph::Node n6("n6"); + + // Directed + graph::DirectedGraph dg; + dg.addEdge(graph::DirectedEdge(n1, n2), 2); + dg.addEdge(graph::DirectedEdge(n1, n2, "multi"), 1); + dg.addEdge(graph::DirectedEdge(n1, n3), 2); + dg.addEdge(graph::DirectedEdge(n1, n4), 1); + dg.addEdge(graph::DirectedEdge(n1, n5), 6); + dg.addEdge(graph::DirectedEdge(n2, n5), 3); + dg.addEdge(graph::DirectedEdge(n5, n6), 2); + dg.addEdge(graph::DirectedEdge(n6, n1), 1); + + res = graph::shortestpath::Dijkstra::dijkstra(dg, n1); + + CPPUNIT_ASSERT_EQUAL(0, res[n1]); + CPPUNIT_ASSERT_EQUAL(1, res[n2]); + CPPUNIT_ASSERT_EQUAL(2, res[n3]); + CPPUNIT_ASSERT_EQUAL(1, res[n4]); + CPPUNIT_ASSERT_EQUAL(4, res[n5]); + CPPUNIT_ASSERT_EQUAL(6, res[n6]); + + // Undirected + graph::UndirectedGraph ug; + ug.addEdge(graph::UndirectedEdge(n1, n2), 2); + ug.addEdge(graph::UndirectedEdge(n1, n2, "multi"), 1); + ug.addEdge(graph::UndirectedEdge(n1, n3), 2); + ug.addEdge(graph::UndirectedEdge(n1, n4), 1); + ug.addEdge(graph::UndirectedEdge(n1, n5), 6); + ug.addEdge(graph::UndirectedEdge(n2, n5), 3); + ug.addEdge(graph::UndirectedEdge(n5, n6), 2); + + res = graph::shortestpath::Dijkstra::dijkstra(ug, n1); + + CPPUNIT_ASSERT_EQUAL(0, res[n1]); + CPPUNIT_ASSERT_EQUAL(1, res[n2]); + CPPUNIT_ASSERT_EQUAL(2, res[n3]); + CPPUNIT_ASSERT_EQUAL(1, res[n4]); + CPPUNIT_ASSERT_EQUAL(4, res[n5]); + CPPUNIT_ASSERT_EQUAL(6, res[n6]); +} + +void GraphDijkstraTest::testNegativeValue() +{ + // Common + bool exception; + graph::shortestpath::Dijkstra::Result res; + graph::Node n1("n1"); + graph::Node n2("n2"); + + // Directed + graph::DirectedGraph dg; + dg.addEdge(graph::DirectedEdge(n1, n2), -2); + + exception = false; + try { + res = graph::shortestpath::Dijkstra::dijkstra(dg, n1); + } catch (exception::AlibException) { + exception = true; + } + + CPPUNIT_ASSERT_EQUAL(true, exception); + + // Undirected + graph::UndirectedGraph ug; + ug.addEdge(graph::UndirectedEdge(n1, n2), -2); + + exception = false; + try { + res = graph::shortestpath::Dijkstra::dijkstra(ug, n1); + } catch (exception::AlibException) { + exception = true; + } + + CPPUNIT_ASSERT_EQUAL(true, exception); +} diff --git a/alib2algo/test-src/graph/shortestpath/DijkstraTest.h b/alib2algo/test-src/graph/shortestpath/DijkstraTest.h new file mode 100644 index 0000000000..d6ddfb568c --- /dev/null +++ b/alib2algo/test-src/graph/shortestpath/DijkstraTest.h @@ -0,0 +1,22 @@ +#ifndef DIJKSTRA_TEST_H_ +#define DIJKSTRA_TEST_H_ + +#include <cppunit/extensions/HelperMacros.h> + +class GraphDijkstraTest : public CppUnit::TestFixture +{ + CPPUNIT_TEST_SUITE(GraphDijkstraTest); + CPPUNIT_TEST(testSimple); + CPPUNIT_TEST(testCycle); + CPPUNIT_TEST(testMultiEdge); + CPPUNIT_TEST(testNegativeValue); + CPPUNIT_TEST_SUITE_END(); + +public: + void testSimple(); + void testCycle(); + void testMultiEdge(); + void testNegativeValue(); +}; + +#endif // DIJKSTRA_TEST_H_ -- GitLab