Renaming Graph to GraphWD, adding tests.

for/release
dmatetelki 11 years ago
parent 5a12ecf6cf
commit 31806b22b9

@ -1,5 +1,5 @@
#ifndef GRAPH_H #ifndef GRAPHWD_HPP
#define GRAPH_H #define GRAPHWD_HPP
#include <unordered_map> #include <unordered_map>
#include <vector> #include <vector>
@ -7,10 +7,11 @@
#include <algorithm> #include <algorithm>
// weighed, directed
template <typename V, template <typename V,
typename E = int> typename E = int>
class Graph { class GraphWD {
public: public:
@ -25,12 +26,12 @@ public:
typedef E weight_type; typedef E weight_type;
typedef const E& const_weight_reference; typedef const E& const_weight_reference;
class vertex_iterator;
class edge_iterator;
private: private:
struct EdgeTo; struct EdgeTo;
// @todo switch to std::unordered_map<V, std::pair<std::vector<V>, std::vector<E>>> for quicker neighbours & weights
// also turning the graph into unordered, weighted, with no multi & self edges by default
typedef std::unordered_map<V, std::vector<EdgeTo> > v_container; typedef std::unordered_map<V, std::vector<EdgeTo> > v_container;
typedef typename v_container::iterator v_iterator; typedef typename v_container::iterator v_iterator;
typedef typename v_container::const_iterator v_const_iterator; typedef typename v_container::const_iterator v_const_iterator;
@ -70,15 +71,14 @@ public:
typedef Edge* edge_pointer; typedef Edge* edge_pointer;
typedef Edge& edge_reference; typedef Edge& edge_reference;
// public:
Graph(bool isdirected = true) : m_directed(isdirected), m_vertices() {} GraphWD(bool isdirected = true) : m_directed(isdirected), m_vertices() {}
Graph(const Graph<V, E>& o) : m_directed(o.m_directed), m_vertices(o.m_vertices) {} GraphWD(const GraphWD<V, E>& o) : m_directed(o.m_directed), m_vertices(o.m_vertices) {}
Graph(std::initializer_list<V> vertex_list); GraphWD(std::initializer_list<V> vertex_list);
Graph(std::initializer_list<Edge> edge_list); GraphWD(std::initializer_list<Edge> edge_list);
Graph<V, E>& operator=(Graph<V, E> o) { swap(o); return *this; } GraphWD<V, E>& operator=(GraphWD<V, E> o) { swap(o); return *this; }
void swap(Graph& o) { std::swap (m_directed, o.m_directed); std::swap(m_vertices, o.m_vertices); } void swap(GraphWD& o) { std::swap (m_directed, o.m_directed); std::swap(m_vertices, o.m_vertices); }
// Properties // Properties
bool directed() const { return m_directed; } bool directed() const { return m_directed; }
@ -108,14 +108,13 @@ public:
// iterators // iterators
/// @todo make it random access
class vertex_iterator : public std::iterator<std::forward_iterator_tag, class vertex_iterator : public std::iterator<std::forward_iterator_tag,
value_type, value_type,
difference_type, difference_type,
pointer, pointer,
reference> reference>
{ {
friend class Graph; friend class GraphWD;
public: public:
typedef vertex_iterator self_type; typedef vertex_iterator self_type;
@ -144,17 +143,20 @@ public:
v_const_iterator m_it; v_const_iterator m_it;
}; };
vertex_iterator vertex_begin() { return vertex_iterator(m_vertices.begin()); } typedef vertex_iterator iterator;
const vertex_iterator vertex_begin() const { return vertex_iterator(m_vertices.begin()); } typedef const vertex_iterator const_iterator;
vertex_iterator vertex_end() { return vertex_iterator(m_vertices.end()); }
const vertex_iterator vertex_end() const { return vertex_iterator(m_vertices.end()); } iterator begin() { return iterator(m_vertices.begin()); }
const_iterator cbegin() const { return const_iterator(m_vertices.begin()); }
iterator end() { return iterator(m_vertices.end()); }
const_iterator cend() const { return const_iterator(m_vertices.end()); }
private: private:
static void eraseEdge(typename std::vector<EdgeTo>& v, const_reference data); static void eraseEdge(typename std::vector<EdgeTo>& v, const_reference data);
static void eraseEdge(typename std::vector<EdgeTo>& v, const_reference data, const_weight_reference weight); static void eraseEdge(typename std::vector<EdgeTo>& v, const_reference data, const_weight_reference weight);
const bool m_directed; bool m_directed;
v_container m_vertices; v_container m_vertices;
}; };
@ -162,21 +164,21 @@ private:
// Edge // Edge
template <typename V, typename E> template <typename V, typename E>
inline Graph<V, E>::Edge::Edge(const_reference s, const_reference d, const_weight_reference w) inline GraphWD<V, E>::Edge::Edge(const_reference s, const_reference d, const_weight_reference w)
: source(s) : source(s)
, destination(d) , destination(d)
, weight(w) , weight(w)
{} {}
template <typename V, typename E> template <typename V, typename E>
inline Graph<V, E>::Edge::Edge(const Edge& o) inline GraphWD<V, E>::Edge::Edge(const Edge& o)
: source(o.source) : source(o.source)
, destination(o.destination) , destination(o.destination)
, weight(o.weight) , weight(o.weight)
{} {}
template <typename V, typename E> template <typename V, typename E>
inline void Graph<V, E>::Edge::swap(Edge& o) inline void GraphWD<V, E>::Edge::swap(Edge& o)
{ {
std::swap(source, o.source); std::swap(source, o.source);
std::swap(destination, o.destination); std::swap(destination, o.destination);
@ -186,45 +188,45 @@ inline void Graph<V, E>::Edge::swap(Edge& o)
// EdgeTo // EdgeTo
template <typename V, typename E> template <typename V, typename E>
inline Graph<V, E>::EdgeTo::EdgeTo(v_iterator destination, const_weight_reference weight) inline GraphWD<V, E>::EdgeTo::EdgeTo(v_iterator destination, const_weight_reference weight)
: m_destination(destination) : m_destination(destination)
, m_weight(weight) , m_weight(weight)
{} {}
template <typename V, typename E> template <typename V, typename E>
inline void Graph<V, E>::EdgeTo::swap(EdgeTo& o) inline void GraphWD<V, E>::EdgeTo::swap(EdgeTo& o)
{ {
std::swap(m_destination, o.m_destination); std::swap(m_destination, o.m_destination);
std::swap(m_weight, o.m_weight); std::swap(m_weight, o.m_weight);
} }
template <typename V, typename E> template <typename V, typename E>
inline bool Graph<V, E>::EdgeTo::operator==(const EdgeTo& other) const inline bool GraphWD<V, E>::EdgeTo::operator==(const EdgeTo& other) const
{ {
return m_destination == other.m_destination && return m_destination == other.m_destination &&
m_weight == other.m_weight; m_weight == other.m_weight;
} }
// Graph // GraphWD
template <typename V, typename E> template <typename V, typename E>
Graph<V, E>::Graph(std::initializer_list<V> vertex_list) GraphWD<V, E>::GraphWD(std::initializer_list<V> vertex_list)
: Graph<V, E>() : GraphWD<V, E>()
{ {
for(const V& v : vertex_list) for(const V& v : vertex_list)
addVertex(v); addVertex(v);
} }
template <typename V, typename E> template <typename V, typename E>
Graph<V, E>::Graph(std::initializer_list<Edge> edge_list) GraphWD<V, E>::GraphWD(std::initializer_list<Edge> edge_list)
: Graph<V, E>() : GraphWD<V, E>()
{ {
for (const Edge& e : edge_list ) for (const Edge& e : edge_list )
addEdge(e.source, e.destination, e.weight); addEdge(e.source, e.destination, e.weight);
} }
template <typename V, typename E> template <typename V, typename E>
inline typename Graph<V, E>::size_type Graph<V, E>::numberOfEdges() const inline typename GraphWD<V, E>::size_type GraphWD<V, E>::numberOfEdges() const
{ {
int sum = 0; int sum = 0;
for (const auto& v : m_vertices) for (const auto& v : m_vertices)
@ -234,17 +236,17 @@ inline typename Graph<V, E>::size_type Graph<V, E>::numberOfEdges() const
} }
template <typename V, typename E> template <typename V, typename E>
inline void Graph<V, E>::addVertex(const_reference data) inline void GraphWD<V, E>::addVertex(const_reference data)
{ {
if (m_vertices.find(data) != m_vertices.end()) if (m_vertices.find(data) != m_vertices.end())
return; return;
std::pair<V, std::vector<Graph<V, E>::EdgeTo> > p(data, std::vector<EdgeTo>()); std::pair<V, std::vector<GraphWD<V, E>::EdgeTo> > p(data, std::vector<EdgeTo>());
m_vertices.insert(p); m_vertices.insert(p);
} }
template <typename V, typename E> template <typename V, typename E>
inline void Graph<V, E>::removeVertex(const_reference data) inline void GraphWD<V, E>::removeVertex(const_reference data)
{ {
v_iterator it = m_vertices.find(data); v_iterator it = m_vertices.find(data);
if (it == m_vertices.end()) if (it == m_vertices.end())
@ -261,7 +263,7 @@ inline void Graph<V, E>::removeVertex(const_reference data)
} }
template <typename V, typename E> template <typename V, typename E>
void Graph<V, E>::addEdge(const_reference source, const_reference destination, const_weight_reference weight) void GraphWD<V, E>::addEdge(const_reference source, const_reference destination, const_weight_reference weight)
{ {
addVertex(source); addVertex(source);
addVertex(destination); addVertex(destination);
@ -269,13 +271,13 @@ void Graph<V, E>::addEdge(const_reference source, const_reference destination, c
v_iterator source_it = m_vertices.find(source); v_iterator source_it = m_vertices.find(source);
v_iterator destination_it = m_vertices.find(destination); v_iterator destination_it = m_vertices.find(destination);
source_it->second.push_back(Graph<V, E>::EdgeTo(destination_it, weight)); source_it->second.push_back(GraphWD<V, E>::EdgeTo(destination_it, weight));
if (!m_directed && source != destination) if (!m_directed && source != destination)
destination_it->second.push_back(Graph<V, E>::EdgeTo(source_it, weight)); destination_it->second.push_back(GraphWD<V, E>::EdgeTo(source_it, weight));
} }
template <typename V, typename E> template <typename V, typename E>
inline void Graph<V, E>::removeEdge(const_reference source, const_reference destination, const_weight_reference weight) inline void GraphWD<V, E>::removeEdge(const_reference source, const_reference destination, const_weight_reference weight)
{ {
v_iterator source_it = m_vertices.find(source); v_iterator source_it = m_vertices.find(source);
if (source_it == m_vertices.end()) if (source_it == m_vertices.end())
@ -291,7 +293,7 @@ inline void Graph<V, E>::removeEdge(const_reference source, const_reference dest
} }
template <typename V, typename E> template <typename V, typename E>
inline void Graph<V, E>::removeEdges(const_reference source, const_reference destination) inline void GraphWD<V, E>::removeEdges(const_reference source, const_reference destination)
{ {
v_iterator source_it = m_vertices.find(source); v_iterator source_it = m_vertices.find(source);
if (source_it == m_vertices.end()) if (source_it == m_vertices.end())
@ -307,7 +309,7 @@ inline void Graph<V, E>::removeEdges(const_reference source, const_reference des
} }
template <typename V, typename E> template <typename V, typename E>
inline std::vector<typename Graph<V, E>::value_type> Graph<V, E>::vertices() const inline std::vector<typename GraphWD<V, E>::value_type> GraphWD<V, E>::vertices() const
{ {
std::vector<value_type> retval; std::vector<value_type> retval;
for (const auto& v : m_vertices) for (const auto& v : m_vertices)
@ -317,7 +319,7 @@ inline std::vector<typename Graph<V, E>::value_type> Graph<V, E>::vertices() con
} }
template <typename V, typename E> template <typename V, typename E>
std::vector<typename Graph<V, E>::value_type> Graph<V, E>::neighboursOf(const_reference data) const std::vector<typename GraphWD<V, E>::value_type> GraphWD<V, E>::neighboursOf(const_reference data) const
{ {
typename std::vector<value_type> retval; typename std::vector<value_type> retval;
v_const_iterator vertex_it = m_vertices.find(data); v_const_iterator vertex_it = m_vertices.find(data);
@ -333,7 +335,7 @@ std::vector<typename Graph<V, E>::value_type> Graph<V, E>::neighboursOf(const_re
} }
template <typename V, typename E> template <typename V, typename E>
std::vector<E> Graph<V, E>::weights(const_reference source, const_reference destination) const std::vector<E> GraphWD<V, E>::weights(const_reference source, const_reference destination) const
{ {
std::vector<E> retval; std::vector<E> retval;
v_const_iterator vertex_it = m_vertices.find(source); v_const_iterator vertex_it = m_vertices.find(source);
@ -351,28 +353,28 @@ std::vector<E> Graph<V, E>::weights(const_reference source, const_reference dest
} }
template <typename V, typename E> template <typename V, typename E>
inline std::vector<typename Graph<V, E>::Edge> Graph<V, E>::edges() const inline std::vector<typename GraphWD<V, E>::Edge> GraphWD<V, E>::edges() const
{ {
std::vector<typename Graph<V, E>::Edge> retval; std::vector<typename GraphWD<V, E>::Edge> retval;
for (const auto& v : m_vertices) for (const auto& v : m_vertices)
for (const auto& e : v.second) for (const auto& e : v.second)
retval.push_back(Graph<V, E>::Edge(v.first, (e.m_destination)->first, e.m_weight)); retval.push_back(GraphWD<V, E>::Edge(v.first, (e.m_destination)->first, e.m_weight));
return retval; return retval;
} }
template <typename V, typename E> template <typename V, typename E>
void Graph<V, E>::eraseEdge(typename std::vector<EdgeTo>& v, const_reference data) { void GraphWD<V, E>::eraseEdge(typename std::vector<EdgeTo>& v, const_reference data) {
v.erase(std::remove_if(v.begin(), v.end(), v.erase(std::remove_if(v.begin(), v.end(),
[&data](const EdgeTo& e) { return e.m_destination->first == data; } ), [&data](const EdgeTo& e) { return e.m_destination->first == data; } ),
v.end()); v.end());
} }
template <typename V, typename E> template <typename V, typename E>
void Graph<V, E>::eraseEdge(typename std::vector<EdgeTo>& v, const_reference data, const_weight_reference weight ) { void GraphWD<V, E>::eraseEdge(typename std::vector<EdgeTo>& v, const_reference data, const_weight_reference weight ) {
v.erase(std::remove_if(v.begin(), v.end(), v.erase(std::remove_if(v.begin(), v.end(),
[&data, &weight](const EdgeTo& e) { return e.m_destination->first == data && e.m_weight == weight; } ), [&data, &weight](const EdgeTo& e) { return e.m_destination->first == data && e.m_weight == weight; } ),
v.end()); v.end());
} }
#endif // GRAPH_H #endif // GRAPHWD_HPP

@ -1,8 +1,30 @@
# set(CMAKE_CXX_FLAGS " -Wall -Wextra -pedantic -Weffc++ -Wshadow --std=c++0x -ggdb")
# set(CMAKE_CXX_COMPILER "/usr/lib/colorgcc/bin/g++")
# set(CMAKE_CXX_COMPILER "/usr/bin/g++-4.8.2")
# set(CMAKE_CXX_COMPILER "/home/denes/bin/colorgcc/g++")
set (CXX_FLAGS "-Wall -Wextra -pedantic -Weffc++ -Wshadow "
# "-Wpointer-arith -Wcast-qual
"-std=c++11 "
"-ggdb -fprofile-arcs -ftest-coverage "
"-fopenmp -D_GLIBCXX_PARALLEL"
)
add_definitions( ${CXX_FLAGS} )
set (CMAKE_CXX_COMPILER "/usr/bin/g++-4.9.0")
find_package(CxxTest) find_package(CxxTest)
# if ( NOT CXXTEST_FOUND )
# message(FATAL_ERROR "Package CXXTEST required, but not found!")
# else()
if(CXXTEST_FOUND) if(CXXTEST_FOUND)
include_directories(${CXXTEST_INCLUDE_DIR} include_directories(${CXXTEST_INCLUDE_DIR}
../lib) ../lib
/usr/include/libxml2)
enable_testing() enable_testing()
@ -10,8 +32,16 @@ if(CXXTEST_FOUND)
generated_main.cpp generated_main.cpp
graph/test_graph.hpp) graph/test_graph.hpp)
target_link_libraries(test_graph
gcov
gomp
xml2
)
endif() endif()
add_custom_target( test add_custom_target( test
COMMAND ./test_graph DEPENDS test_graph
COMMAND ./tools/run_test.sh ./test_graph
) )

@ -1,6 +1,24 @@
#include <cxxtest/TestSuite.h> #include <cxxtest/TestSuite.h>
#include "graph/graph.hpp" #include <graph/graph.hpp>
// #include <graph/graph_algorithms.hpp>
// #include <graph/mesh_algorithms.hpp>
// #include <graph/graph_xml.hpp>
// #include <graph/common.hpp>
namespace std {
template <>
struct hash<std::pair<float,float> >
{
std::size_t operator()(const std::pair<float,float>& k) const {
std::size_t h1 = std::hash<float>()(k.first);
std::size_t h2 = std::hash<float>()(k.second);
return h1 ^ (h2 << 1);
}
};
}
namespace test { class TestSubsystem1DummyClassSuite; } namespace test { class TestSubsystem1DummyClassSuite; }
@ -10,149 +28,456 @@ class test::TestSubsystem1DummyClassSuite : public CxxTest::TestSuite
public: public:
void testBasic() void test_trivial()
{
GraphWD<std::string> g(false);
TS_ASSERT_EQUALS ( g.directed(), false );
GraphWD<int> g1;
TS_ASSERT_EQUALS ( g1.directed(), true );
GraphWD<float, std::string> g2;
TS_ASSERT_EQUALS ( g2.empty(), true );
GraphWD<std::pair<float, float> ,float> g3;
TS_ASSERT_EQUALS ( g3.numberOfVertices(), 0 );
TS_ASSERT_EQUALS ( g3.numberOfEdges(), 0 );
TS_ASSERT_EQUALS ( g3.contains(std::make_pair(2.4, 5.567)), false );
TS_ASSERT_EQUALS ( g3.vertices().size(), 0 );
TS_ASSERT_EQUALS ( g3.edges().size(), 0 );
TS_ASSERT_EQUALS ( g3.empty(), true );
}
void test_basic_vertices()
{ {
GraphWD<std::string> g;
Graph<int> g; g.addVertex("a");
g.addVertex("b");
g.addVertex("c");
TS_ASSERT_EQUALS ( g.vertices().size(), 3 );
TS_ASSERT_EQUALS(g.empty(), true); g.addVertex("d");
TS_ASSERT_EQUALS(g.numberOfVertices(), 0); g.addVertex("d");
TS_ASSERT_EQUALS(g.numberOfEdges(), 0); TS_ASSERT_EQUALS ( g.vertices().size(), 4 );
g.removeVertex("f");
TS_ASSERT_EQUALS ( g.vertices().size(), 4 );
int a = 2; g.removeVertex("c");
int b = 5; TS_ASSERT_EQUALS ( g.vertices().size(), 3 );
TS_ASSERT_EQUALS(g.addEdge(a, b), false); g.removeVertex("c");
TS_ASSERT_EQUALS ( g.vertices().size(), 3 );
TS_ASSERT_EQUALS(g.addVertex(a), true); for(const std::string& s : g.vertices())
TS_ASSERT_EQUALS(g.addVertex(a), false); g.removeVertex(s);
TS_ASSERT_EQUALS(g.empty(), false);
TS_ASSERT_EQUALS(g.numberOfVertices(), 1);
{
const std::vector<const int*> v = g.vertices();
TS_ASSERT_EQUALS(v.size(), 1);
TS_ASSERT_EQUALS(*(v[0]), 2);
}
TS_ASSERT_EQUALS(g.addEdge(a, b), false); TS_ASSERT_EQUALS ( g.vertices().size(), 0 );
TS_ASSERT_EQUALS(g.addVertex(b), true); GraphWD<std::string> g2 = { "a", "b", "c" };
TS_ASSERT_EQUALS(g.numberOfVertices(), 2); TS_ASSERT_EQUALS ( g2.vertices().size(), 3 );
}
void test_basic_edges()
{ {
const std::vector<const int*> v = g.vertices(); GraphWD<std::string> g = { "a", "a", "b", "c", "d", "e" };
TS_ASSERT_EQUALS(v.size(), 2); g.addEdge("a", "a", 1);
TS_ASSERT_EQUALS(*(v[0]), 2); g.addEdge("a", "b", 2);
TS_ASSERT_EQUALS(*(v[1]), 5); g.addEdge("a", "c", 3);
g.addEdge("c", "d", 4);
TS_ASSERT_EQUALS ( g.edges().size(), 4 );
TS_ASSERT_EQUALS ( g.neighboursOf("a").size(), 3 );
TS_ASSERT_EQUALS ( g.neighboursOf("b").size(), 0 );
TS_ASSERT_EQUALS ( g.neighboursOf("c").size(), 1 );
TS_ASSERT_EQUALS ( g.neighboursOf("d").size(), 0 );
TS_ASSERT_EQUALS ( g.neighboursOf("e").size(), 0 );
std::vector<std::string> n = g.neighboursOf("a");
TS_ASSERT_DIFFERS ( find(n.begin(), n.end(), "a"), n.end() );
TS_ASSERT_DIFFERS ( find(n.begin(), n.end(), "b"), n.end() );
TS_ASSERT_DIFFERS ( find(n.begin(), n.end(), "c"), n.end() );
TS_ASSERT_EQUALS ( find(n.begin(), n.end(), "d"), n.end() );
TS_ASSERT_EQUALS ( find(n.begin(), n.end(), "e"), n.end() );
std::vector<int> w = g.weights("a", "b");
TS_ASSERT_EQUALS ( w.size(), 1 );
TS_ASSERT_EQUALS ( w[0], 2 );
// undirected
GraphWD<std::string> g1(false);
g1.addVertex("a");
g1.addVertex("b");
g1.addVertex("c");
g1.addVertex("d");
g1.addVertex("e");
g1.addEdge("a", "a", 1);
g1.addEdge("a", "b", 2);
g1.addEdge("a", "c", 3);
g1.addEdge("c", "d", 4);
TS_ASSERT_EQUALS ( g1.edges().size(), 4*2 - 1 );
TS_ASSERT_EQUALS ( g1.neighboursOf("a").size(), 3 );
TS_ASSERT_EQUALS ( g1.neighboursOf("b").size(), 1 );
TS_ASSERT_EQUALS ( g1.neighboursOf("c").size(), 2 );
TS_ASSERT_EQUALS ( g1.neighboursOf("d").size(), 1 );
TS_ASSERT_EQUALS ( g1.neighboursOf("e").size(), 0 );
} }
TS_ASSERT_EQUALS(g.numberOfEdges(), 0); void test_list_ctor_from_vertices()
TS_ASSERT_EQUALS(g.addEdge(a, b), true); {
GraphWD<std::string> g = { "a", "a", "b", "c", "d", "e" };
TS_ASSERT_EQUALS(g.numberOfEdges(), 1); std::vector<std::string> v = g.vertices();
TS_ASSERT_EQUALS ( v.size(), 5 );
}
void test_move_ctor()
{ {
const std::vector<int> eb = g.weightsBetween(2, 5);
TS_ASSERT_EQUALS(eb.size(), 1); GraphWD<std::string> g = {
TS_ASSERT_EQUALS(eb[0], 0); {"a", "a", 1},
{"a", "b", 2},
{"a", "c", 3},
{"c", "d", 4} };
GraphWD<std::string> g1;
g1 = g;
TS_ASSERT_EQUALS ( g.edges().size(), 4 );
TS_ASSERT_EQUALS ( g.neighboursOf("a").size(), 3 );
TS_ASSERT_EQUALS ( g.neighboursOf("b").size(), 0 );
TS_ASSERT_EQUALS ( g.neighboursOf("c").size(), 1 );
TS_ASSERT_EQUALS ( g.neighboursOf("d").size(), 0 );
TS_ASSERT_EQUALS ( g1.edges().size(), 4 );
TS_ASSERT_EQUALS ( g1.neighboursOf("a").size(), 3 );
TS_ASSERT_EQUALS ( g1.neighboursOf("b").size(), 0 );
TS_ASSERT_EQUALS ( g1.neighboursOf("c").size(), 1 );
TS_ASSERT_EQUALS ( g1.neighboursOf("d").size(), 0 );
} }
TS_ASSERT_EQUALS(g.weightsBetween(5, 2).size(), 0);
TS_ASSERT_EQUALS(g.neighboursOf(2).size(), 1); void test_list_ctor_from_edges()
{ {
const std::vector<const int*> n = g.neighboursOf(2);
TS_ASSERT_EQUALS(n.size(), 1); GraphWD<std::string> g = {
TS_ASSERT_EQUALS(*(n[0]), 5); {"a", "a", 1},
{"a", "b", 2},
{"a", "c", 3},
{"c", "d", 4} };
TS_ASSERT_EQUALS ( g.vertices().size(), 4 );
TS_ASSERT_EQUALS ( g.edges().size(), 4 );
TS_ASSERT_EQUALS ( g.neighboursOf("a").size(), 3 );
TS_ASSERT_EQUALS ( g.neighboursOf("b").size(), 0 );
TS_ASSERT_EQUALS ( g.neighboursOf("c").size(), 1 );
TS_ASSERT_EQUALS ( g.neighboursOf("d").size(), 0 );
TS_ASSERT_EQUALS ( g.neighboursOf("e").size(), 0 );
} }
int c = 13; void test_adding_and_removing_vertices()
int d = 1; {
TS_ASSERT_EQUALS(g.numberOfVertices(), 2); GraphWD<std::string> g;
g.addVertex(c); g.addVertex("a");
g.addVertex(d); g.addVertex("b");
TS_ASSERT_EQUALS(g.numberOfVertices(), 4); g.addVertex("c");
int vertices_array[] = {2, 5, 13, 1}; g.addVertex("d");
std::vector<int> v(vertices_array, vertices_array + sizeof(vertices_array) / sizeof(int) ); g.addVertex("e");
Graph<int>::vertex_iterator g_it2; g.addVertex("f");
std::vector<int>::iterator v_it = v.begin(); g.addVertex("g");
g_it2 = g.vertex_begin();
TS_ASSERT_EQUALS(g_it2, g.vertex_begin()); g.addEdge("a", "a", 1);
g.addEdge("a", "b", 2);
for (Graph<int>::vertex_iterator g_it = g.vertex_begin(); g.addEdge("a", "c", 3);
g_it != g.vertex_end(); g.addEdge("c", "d", 4);
++g_it, ++v_it) { g.addEdge("c", "f", 5);
TS_ASSERT_EQUALS(*g_it, *v_it); g.addEdge("f", "e", 6);
TS_ASSERT_EQUALS ( g.vertices().size(), 7 );
TS_ASSERT_EQUALS ( g.edges().size(), 6 );
g.removeVertex("c");
TS_ASSERT_EQUALS ( g.vertices().size(), 7-1 );
TS_ASSERT_EQUALS ( g.edges().size(), 6-3 );
std::vector<GraphWD<std::string>::Edge> es = g.edges();
// TS_ASSERT_EQUALS ( es[0].source, "a" );
// TS_ASSERT_EQUALS ( es[0].destination, "a" );
// TS_ASSERT_EQUALS ( es[0].weight, 1 );
//
// TS_ASSERT_EQUALS ( es[1].source, "a" );
// TS_ASSERT_EQUALS ( es[1].destination, "b" );
// TS_ASSERT_EQUALS ( es[1].weight, 2 );
//
// TS_ASSERT_EQUALS ( es[2].source, "f" );
// TS_ASSERT_EQUALS ( es[2].destination, "e" );
// TS_ASSERT_EQUALS ( es[2].weight, 6 );
} }
void test_removing_edges()
{
TS_ASSERT_EQUALS(g.neighboursOf(5).size(), 0); GraphWD<std::string> g;
g.addVertex("a");
Graph<int> g2; g.addVertex("b");
g2.addVertex(1); g.addVertex("c");
g2.addVertex(2); g.addVertex("d");
g2.addVertex(3); g.addVertex("e");
g2.addVertex(4); g.addVertex("f");
g.addVertex("g");
TS_ASSERT_EQUALS(g2.addEdge(1, 2), true); g.addEdge("a", "a", 1);
TS_ASSERT_EQUALS(g2.addEdge(1, 3), true); g.addEdge("a", "b", 21); // multiple to b
TS_ASSERT_EQUALS(g2.addEdge(2, 4), true); g.addEdge("a", "b", 22);
std::vector<Graph<int>::Edge> e = g2.edges(); g.addEdge("a", "b", 23);
TS_ASSERT_EQUALS(e.size(), 3); g.addEdge("a", "c", 3);
g.addEdge("c", "d", 4);
Graph<int>::edge_iterator edge_it; g.addEdge("c", "f", 5);
Graph<int>::edge_iterator beee = g2.edge_begin(); g.addEdge("f", "e", 6);
edge_it = beee;
// TS_ASSERT_EQUALS(edge_it, g2.edge_begin()); g.removeEdge("c", "d", 4);
int source = (*edge_it).getSource(); g.removeEdge("e", "f", 1234);
int destination = (*edge_it).getDestination();
TS_ASSERT_EQUALS (source, 1); g.removeEdges("a", "b");
TS_ASSERT_EQUALS (destination, 2); g.removeEdges("invalid", "b");
++edge_it; TS_ASSERT_EQUALS( g.vertices().size(), 7 );
source = (*edge_it).getSource(); TS_ASSERT_EQUALS ( g.edges().size(), 4 );
destination = (*edge_it).getDestination();
TS_ASSERT_EQUALS (source, 1); // int j; GraphWD<std::string>::edge_iterator e_it;
TS_ASSERT_EQUALS (destination, 3); // for (j = 0, e_it = g.edge_begin();
// e_it != g.edge_end();
++edge_it; // ++e_it, ++j)
source = (*edge_it).getSource(); // switch (j) {
destination = (*edge_it).getDestination(); // case 0: TS_ASSERT_EQUALS (e_it->source, "a");
TS_ASSERT_EQUALS (source, 2); // TS_ASSERT_EQUALS (e_it->destination, "a");
TS_ASSERT_EQUALS (destination, 4); // TS_ASSERT_EQUALS (e_it->weight, 1);
// break;
int aasdads = 23; // case 1: TS_ASSERT_EQUALS (e_it->source, "a");
// Graph<int>::edge_iterator edge_it; // TS_ASSERT_EQUALS (e_it->destination, "c");
// TS_ASSERT_EQUALS (e_it->weight, 3);
for(/*Graph<int>::edge_iterator*/ edge_it = g2.edge_begin(); edge_it != g2.edge_end(); ++edge_it) { // break;
int source = (*edge_it).getSource(); // case 2: TS_ASSERT_EQUALS (e_it->source, "c");
int destination = (*edge_it).getDestination(); // TS_ASSERT_EQUALS (e_it->destination, "f");
// std::cout << "edge: " // TS_ASSERT_EQUALS (e_it->weight, 5);
// << source << " " // break;
// << destination << " " // case 3: TS_ASSERT_EQUALS (e_it->source, "f");
// << (*edge_it).getWeight() << std::endl; // TS_ASSERT_EQUALS (e_it->destination, "e");
// TS_ASSERT_EQUALS (e_it->weight, 6);
// break;
// }
} }
Graph<int>::edge_iterator edge_it1 = g2.edge_begin(); void test_iterator()
edge_it1 += 10; {
// TS_ASSERT_EQUALS(edge_it1, g2.edge_end());
Graph<int>::edge_iterator edge_it2 = g2.edge_begin(); GraphWD<std::string> g;
edge_it2 += 2; g.addVertex("a");
TS_ASSERT_EQUALS( (*edge_it2).getSource(), 2); g.addVertex("b");
TS_ASSERT_EQUALS( (*edge_it2).getDestination(), 4); g.addVertex("c");
TS_ASSERT_EQUALS((*edge_it2).getWeight(), 0); g.addVertex("d");
g.addVertex("e");
g.addVertex("f");
g.addVertex("g");
g.addEdge("a", "a", 1);
g.addEdge("a", "b", 2);
g.addEdge("a", "c", 3);
g.addEdge("c", "d", 4);
g.addEdge("c", "f", 5);
g.addEdge("f", "e", 6);
// int i; GraphWD<std::string>::vertex_iterator v_it;
// for (i = 0, v_it = g.begin();
// v_it != g.end();
// ++v_it, ++i)
// switch (i) {
// case 0: TS_ASSERT_EQUALS (*v_it, "a"); break;
// case 1: TS_ASSERT_EQUALS (*v_it, "b"); break;
// case 2: TS_ASSERT_EQUALS (*v_it, "c"); break;
// case 3: TS_ASSERT_EQUALS (*v_it, "d"); break;
// case 4: TS_ASSERT_EQUALS (*v_it, "e"); break;
// case 5: TS_ASSERT_EQUALS (*v_it, "f"); break;
// case 6: TS_ASSERT_EQUALS (*v_it, "g"); break;
// }
// int j; GraphWD<std::string>::edge_iterator e_it;
// for (j = 0, e_it = g.edge_begin();
// e_it != g.edge_end();
// ++e_it, ++j)
// switch (j) {
// case 0: TS_ASSERT_EQUALS (e_it->source, "a");
// TS_ASSERT_EQUALS (e_it->destination, "a");
// TS_ASSERT_EQUALS (e_it->weight, 1);
// break;
// case 1: TS_ASSERT_EQUALS (e_it->source, "a");
// TS_ASSERT_EQUALS (e_it->destination, "b");
// TS_ASSERT_EQUALS (e_it->weight, 2);
// break;
// case 2: TS_ASSERT_EQUALS (e_it->source, "a");
// TS_ASSERT_EQUALS (e_it->destination, "c");
// TS_ASSERT_EQUALS (e_it->weight, 3);
// break;
// case 3: TS_ASSERT_EQUALS (e_it->source, "c");
// TS_ASSERT_EQUALS (e_it->destination, "d");
// TS_ASSERT_EQUALS (e_it->weight, 4);
// break;
// case 4: TS_ASSERT_EQUALS (e_it->source, "c");
// TS_ASSERT_EQUALS (e_it->destination, "f");
// TS_ASSERT_EQUALS (e_it->weight, 5);
// break;
// case 5: TS_ASSERT_EQUALS (e_it->source, "f");
// TS_ASSERT_EQUALS (e_it->destination, "e");
// TS_ASSERT_EQUALS (e_it->weight, 6);
// break;
// }
// g.removeVertex("c");
//
// for (i = 0, v_it = g.vertex_begin();
// v_it != g.vertex_end();
// ++v_it, ++i)
// switch (i) {
// case 0: TS_ASSERT_EQUALS (*v_it, "a"); break;
// case 1: TS_ASSERT_EQUALS (*v_it, "b"); break;
// case 2: TS_ASSERT_EQUALS (*v_it, "d"); break;
// case 3: TS_ASSERT_EQUALS (*v_it, "e"); break;
// case 4: TS_ASSERT_EQUALS (*v_it, "f"); break;
// case 5: TS_ASSERT_EQUALS (*v_it, "g"); break;
// }
//
// for (j = 0, e_it = g.edge_begin();
// e_it != g.edge_end();
// ++e_it, ++j)
// switch (j) {
// case 0: TS_ASSERT_EQUALS (e_it->source, "a");
// TS_ASSERT_EQUALS (e_it->destination, "a");
// TS_ASSERT_EQUALS (e_it->weight, 1);
// break;
// case 1: TS_ASSERT_EQUALS (e_it->source, "a");
// TS_ASSERT_EQUALS (e_it->destination, "b");
// TS_ASSERT_EQUALS (e_it->weight, 2);
// break;
// case 2: TS_ASSERT_EQUALS (e_it->source, "f");
// TS_ASSERT_EQUALS (e_it->destination, "e");
// TS_ASSERT_EQUALS (e_it->weight, 6);
// break;
// }
}
// toDot(g2, "itt.dot"); void test_wights()
// Graph<int> read_g; {
// fromDot("itt.dot", read_g);
// TS_ASSERT_EQUALS(g2, read_g);
GraphWD<std::string> g;
g.addVertex("a");
g.addVertex("b");
g.addVertex("c");
g.addVertex("d");
g.addVertex("e");
g.addVertex("f");
g.addVertex("g");
g.addEdge("a", "a", 1);
g.addEdge("a", "b", 2);
g.addEdge("a", "c", 3);
g.addEdge("c", "d", 4);
g.addEdge("c", "f", 5);
g.addEdge("f", "e", 6);
TS_ASSERT_EQUALS ( g.weights("a", "a").size(), 1 );
TS_ASSERT_EQUALS ( g.weights("a", "b").size(), 1 );
TS_ASSERT_EQUALS ( g.weights("a", "c").size(), 1 );
TS_ASSERT_EQUALS ( g.weights("c", "d").size(), 1 );
TS_ASSERT_EQUALS ( g.weights("f", "e").size(), 1 );
TS_ASSERT_EQUALS ( g.weights("a", "d").size(), 0 );
TS_ASSERT_EQUALS ( g.weights("a", "f").size(), 0 );
TS_ASSERT_EQUALS ( g.weights("a", "e").size(), 0 );
TS_ASSERT_EQUALS ( g.weights("a", "d").size(), 0 );
TS_ASSERT_EQUALS ( g.weights("d", "e").size(), 0 );
TS_ASSERT_EQUALS ( g.weights("b", "f").size(), 0 );
// directed
TS_ASSERT_EQUALS ( g.weights("a", "a").size(), 1 );
TS_ASSERT_EQUALS ( g.weights("b", "b").size(), 0 );
TS_ASSERT_EQUALS ( g.weights("c", "a").size(), 0 );
TS_ASSERT_EQUALS ( g.weights("d", "c").size(), 0 );
TS_ASSERT_EQUALS ( g.weights("f", "c").size(), 0 );
TS_ASSERT_EQUALS ( g.weights("e", "f").size(), 0 );
}
// /// @todo put algorithm tests to another suite
// void test_dijkstra()
// {
//
// // from http://en.wikipedia.org/wiki/File:Dijkstra_Animation.gif
// GraphWDWD<int> d(false);
// d.addVertex(1);
// d.addVertex(2);
// d.addVertex(3);
// d.addVertex(4);
// d.addVertex(5);
// d.addVertex(6);
//
// d.addVertex(7); // unconnected extra
// d.addVertex(8);
// d.addVertex(9);
//
// d.addEdge(1, 2, 7);
// d.addEdge(1, 3, 9);
// d.addEdge(1, 6, 14);
// d.addEdge(2, 3, 10);
// d.addEdge(2, 4, 15);
// d.addEdge(3, 4, 11);
// d.addEdge(3, 6, 2);
// d.addEdge(4, 5, 6);
// d.addEdge(5, 6, 9);
//
// d.addEdge(8, 9, 9);
//
// std::pair<std::unordered_map< int, int>, std::unordered_map< int, int> > shortest_path = dijkstra_shortest_path_v2(d, 1);
//
// TS_ASSERT_EQUALS ( shortest_path.first[1], 0 );
// TS_ASSERT_EQUALS ( shortest_path.first[2], 7 );
// TS_ASSERT_EQUALS ( shortest_path.first[3], 9 );
// TS_ASSERT_EQUALS ( shortest_path.first[4], 20 );
// TS_ASSERT_EQUALS ( shortest_path.first[5], 20 );
// TS_ASSERT_EQUALS ( shortest_path.first[6], 11 );
// TS_ASSERT_EQUALS ( shortest_path.first.size(), 6 );
//
// TS_ASSERT_EQUALS ( shortest_path.second[2], 1 );
// TS_ASSERT_EQUALS ( shortest_path.second[3], 1 );
// TS_ASSERT_EQUALS ( shortest_path.second[4], 3 );
// TS_ASSERT_EQUALS ( shortest_path.second[5], 6 );
// TS_ASSERT_EQUALS ( shortest_path.second[6], 3 );
// TS_ASSERT_EQUALS ( shortest_path.second.size(), 5 );
// }
//
// /// @todo put XML tests to another suite
// void test_read_from_XML()
// {
//
// GraphWD< std::pair<float, float> > g_xml = readGraphWD<float>("/home/denes/projects/graph/lib/graph/graph_example.xml");
// // for (std::pair<float, float> v : g_xml.vertices())
// // std::cout << TToStr(v) << std::endl;
// //
// // for (GraphWD< std::pair<float, float> >::Edge e : g_xml.edges())
// // std::cout << TToStr(e.source) << " -> " << TToStr(e.destination) << std::endl;
// }
// {
// std::vector< std::pair<float, float> > points = {
// {1, 1},
// };
// GraphWD< std::pair<float, float> > g_d = delaunay_triangulation(points);
// }
Graph<int> g4;
g4.addVertex(3);
Graph<int> g5 = g4;
Graph<int> g6;
g6 = g5;
g6.removeVertex(3);
}
}; };

Loading…
Cancel
Save