// Copyright (C) 2007 Davis E. King (davis@dlib.net) // License: Boost Software License See LICENSE.txt for the full license. #ifndef DLIB_GRAPH_KERNEl_1_ #define DLIB_GRAPH_KERNEl_1_ #include <memory> #include <vector> #include "../serialize.h" #include "../noncopyable.h" #include "../std_allocator.h" #include "../algs.h" #include "graph_kernel_abstract.h" #include "../is_kind.h" namespace dlib { // ---------------------------------------------------------------------------------------- template <typename node_type, typename graph, bool is_checked> struct graph_checker_helper { /*! This object is used to check preconditions based on the value of is_checked !*/ static void check_neighbor ( unsigned long edge_index, const node_type& self ) { // make sure requires clause is not broken DLIB_CASSERT(edge_index < self.number_of_neighbors(), "\tnode_type& graph::node_type::neighbor(edge_index)" << "\n\tYou have specified an invalid index" << "\n\tedge_index: " << edge_index << "\n\tnumber_of_neighbors(): " << self.number_of_neighbors() << "\n\tthis: " << &self ); } static void check_edge ( unsigned long edge_index, const node_type& self ) { // make sure requires clause is not broken DLIB_CASSERT(edge_index < self.number_of_neighbors(), "\tE& graph::node_type::edge(edge_index)" << "\n\tYou have specified an invalid index" << "\n\tedge_index: " << edge_index << "\n\tnumber_of_neighbors(): " << self.number_of_neighbors() << "\n\tthis: " << &self ); } static void check_node ( unsigned long index, const graph& self ) { // make sure requires clause is not broken DLIB_CASSERT(index < self.number_of_nodes(), "\tnode_type& graph::node(index)" << "\n\tYou have specified an invalid index" << "\n\tindex: " << index << "\n\tnumber_of_nodes(): " << self.number_of_nodes() << "\n\tthis: " << &self ); } static void check_has_edge ( unsigned long node_index1, unsigned long node_index2, const graph& self ) { // make sure requires clause is not broken DLIB_CASSERT(node_index1 < self.number_of_nodes() && node_index2 < self.number_of_nodes(), "\tvoid graph::has_edge(node_index1, node_index2)" << "\n\tYou have specified an invalid index" << "\n\tnode_index1: " << node_index1 << "\n\tnode_index2: " << node_index2 << "\n\tnumber_of_nodes(): " << self.number_of_nodes() << "\n\tthis: " << &self ); } static void check_add_edge ( unsigned long node_index1, unsigned long node_index2, const graph& self ) { DLIB_CASSERT(node_index1 < self.number_of_nodes() && node_index2 < self.number_of_nodes(), "\tvoid graph::add_edge(node_index1, node_index2)" << "\n\tYou have specified an invalid index" << "\n\tnode_index1: " << node_index1 << "\n\tnode_index2: " << node_index2 << "\n\tnumber_of_nodes(): " << self.number_of_nodes() << "\n\tthis: " << &self ); DLIB_CASSERT( self.has_edge(node_index1, node_index2) == false, "\tvoid graph::add_edge(node_index1, node_index2)" << "\n\tYou can't add an edge if it already exists in the graph" << "\n\tnode_index1: " << node_index1 << "\n\tnode_index2: " << node_index2 << "\n\tnumber_of_nodes(): " << self.number_of_nodes() << "\n\tthis: " << &self ); } static void check_remove_edge ( unsigned long node_index1, unsigned long node_index2, const graph& self ) { DLIB_CASSERT(node_index1 < self.number_of_nodes() && node_index2 < self.number_of_nodes(), "\tvoid graph::remove_edge(node_index1, node_index2)" << "\n\tYou have specified an invalid index" << "\n\tnode_index1: " << node_index1 << "\n\tnode_index2: " << node_index2 << "\n\tnumber_of_nodes(): " << self.number_of_nodes() << "\n\tthis: " << &self ); DLIB_CASSERT( self.has_edge(node_index1, node_index2) == true, "\tvoid graph::remove_edge(node_index1, node_index2)" << "\n\tYou can't remove an edge if it isn't in the graph" << "\n\tnode_index1: " << node_index1 << "\n\tnode_index2: " << node_index2 << "\n\tnumber_of_nodes(): " << self.number_of_nodes() << "\n\tthis: " << &self ); } static void check_remove_node ( unsigned long index, const graph& self ) { // make sure requires clause is not broken DLIB_CASSERT(index < self.number_of_nodes(), "\tvoid graph::remove_node(index)" << "\n\tYou have specified an invalid index" << "\n\tindex: " << index << "\n\tnumber_of_nodes(): " << self.number_of_nodes() << "\n\tthis: " << &self ); } }; template <typename node_type, typename graph> struct graph_checker_helper <node_type, graph, false> { static inline void check_edge ( unsigned long , const node_type& ) { } static inline void check_neighbor ( unsigned long , const node_type& ) { } static inline void check_node ( unsigned long , const graph& ) { } static inline void check_has_edge ( unsigned long , unsigned long , const graph& ) { } static inline void check_add_edge ( unsigned long , unsigned long , const graph& ) { } static inline void check_remove_edge ( unsigned long , unsigned long , const graph& ) { } static inline void check_remove_node ( unsigned long , const graph& ) { } }; // ---------------------------------------------------------------------------------------- template < typename T, typename E = char, typename mem_manager = default_memory_manager, bool is_checked = true > class graph_kernel_1 : noncopyable { /*! INITIAL VALUE - nodes.size() == 0 CONVENTION - nodes.size() == number_of_nodes() - for all valid i: - *nodes[i] == node(i) - nodes[i]->neighbors.size() == nodes[i]->number_of_neighbors(i) - nodes[i]->idx == i == nodes[i]->index() - for all valid n: - nodes[i]->neighbors[n] == pointer to the n'th parent node of i - *nodes[i]->neighbors[n] == node(i).neighbor(n) - *nodes[i]->edges[n] == node(i).edge(n) !*/ public: struct node_type; private: typedef graph_checker_helper<node_type, graph_kernel_1, is_checked> checker; public: typedef T type; typedef E edge_type; typedef mem_manager mem_manager_type; graph_kernel_1( ) {} virtual ~graph_kernel_1( ) {} void clear( ) { nodes.clear(); } void set_number_of_nodes ( unsigned long new_size ); unsigned long number_of_nodes ( ) const { return nodes.size(); } node_type& node ( unsigned long index ) { checker::check_node(index,*this); return *nodes[index]; } const node_type& node ( unsigned long index ) const { checker::check_node(index,*this); return *nodes[index]; } bool has_edge ( unsigned long node_index1, unsigned long node_index2 ) const; void add_edge ( unsigned long node_index1, unsigned long node_index2 ); void remove_edge ( unsigned long node_index1, unsigned long node_index2 ); unsigned long add_node ( ); void remove_node ( unsigned long index ); void swap ( graph_kernel_1& item ) { nodes.swap(item.nodes); } public: struct node_type { T data; typedef graph_kernel_1 graph_type; unsigned long index( ) const { return idx; } unsigned long number_of_neighbors ( ) const { return neighbors.size(); } const node_type& neighbor ( unsigned long edge_index ) const { checker::check_neighbor(edge_index,*this); return *neighbors[edge_index]; } node_type& neighbor ( unsigned long edge_index ) { checker::check_neighbor(edge_index,*this); return *neighbors[edge_index]; } const E& edge ( unsigned long edge_index ) const { checker::check_edge(edge_index,*this); return *edges[edge_index]; } E& edge ( unsigned long edge_index ) { checker::check_edge(edge_index,*this); return *edges[edge_index]; } private: friend class graph_kernel_1; typedef std_allocator<node_type*,mem_manager> alloc_type; typedef std_allocator<std::shared_ptr<E>,mem_manager> alloc_edge_type; std::vector<node_type*,alloc_type> neighbors; std::vector<std::shared_ptr<E>,alloc_edge_type> edges; unsigned long idx; }; private: typedef std_allocator<std::shared_ptr<node_type>,mem_manager> alloc_type; typedef std::vector<std::shared_ptr<node_type>, alloc_type> vector_type; vector_type nodes; }; // ---------------------------------------------------------------------------------------- template < typename T, typename E, typename mem_manager, bool is_checked > inline void swap ( graph_kernel_1<T,E,mem_manager,is_checked>& a, graph_kernel_1<T,E,mem_manager,is_checked>& b ) { a.swap(b); } // ---------------------------------------------------------------------------------------- template < typename T, typename E, typename mem_manager, bool is_checked > struct is_graph<graph_kernel_1<T,E,mem_manager, is_checked> > { static const bool value = true; }; // ---------------------------------------------------------------------------------------- template < typename T, typename E, typename mem_manager, bool is_checked > void serialize ( const graph_kernel_1<T,E,mem_manager,is_checked>& item, std::ostream& out ) { try { serialize(item.number_of_nodes(), out); // serialize each node for (unsigned long i = 0; i < item.number_of_nodes(); ++i) { serialize(item.node(i).data, out); // serialize all the edges for (unsigned long n = 0; n < item.node(i).number_of_neighbors(); ++n) { // only serialize edges that we haven't already serialized if (item.node(i).neighbor(n).index() >= i) { serialize(item.node(i).neighbor(n).index(), out); serialize(item.node(i).edge(n), out); } } const unsigned long stop_mark = 0xFFFFFFFF; serialize(stop_mark, out); } } catch (serialization_error& e) { throw serialization_error(e.info + "\n while serializing object of type graph_kernel_1"); } } // ---------------------------------------------------------------------------------------- template < typename T, typename E, typename mem_manager, bool is_checked > void deserialize ( graph_kernel_1<T,E,mem_manager,is_checked>& item, std::istream& in ) { try { unsigned long size; deserialize(size, in); item.clear(); item.set_number_of_nodes(size); // deserialize each node for (unsigned long i = 0; i < item.number_of_nodes(); ++i) { deserialize(item.node(i).data, in); const unsigned long stop_mark = 0xFFFFFFFF; // Add all the edges going to this node's neighbors unsigned long index; deserialize(index, in); while (index != stop_mark) { item.add_edge(i, index); // find the edge unsigned long j = 0; for (j = 0; j < item.node(i).number_of_neighbors(); ++j) if (item.node(i).neighbor(j).index() == index) break; deserialize(item.node(i).edge(j), in); deserialize(index, in); } } } catch (serialization_error& e) { throw serialization_error(e.info + "\n while deserializing object of type graph_kernel_1"); } } // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- // member function definitions // ---------------------------------------------------------------------------------------- // ---------------------------------------------------------------------------------------- template < typename T, typename E, typename mem_manager, bool is_checked > void graph_kernel_1<T,E,mem_manager,is_checked>:: set_number_of_nodes ( unsigned long new_size ) { try { nodes.resize(new_size); for (unsigned long i = 0; i < nodes.size(); ++i) { nodes[i].reset(new node_type); nodes[i]->idx = i; } } catch (...) { clear(); throw; } } // ---------------------------------------------------------------------------------------- template < typename T, typename E, typename mem_manager, bool is_checked > bool graph_kernel_1<T,E,mem_manager,is_checked>:: has_edge ( unsigned long node_index1, unsigned long node_index2 ) const { checker::check_has_edge(node_index1, node_index2, *this); node_type& n = *nodes[node_index1]; // search all the child nodes to see if there is a link to the right node for (unsigned long i = 0; i < n.neighbors.size(); ++i) { if (n.neighbors[i]->idx == node_index2) return true; } return false; } // ---------------------------------------------------------------------------------------- template < typename T, typename E, typename mem_manager, bool is_checked > void graph_kernel_1<T,E,mem_manager,is_checked>:: add_edge ( unsigned long node_index1, unsigned long node_index2 ) { checker::check_add_edge(node_index1, node_index2, *this); try { node_type& n1 = *nodes[node_index1]; node_type& n2 = *nodes[node_index2]; n1.neighbors.push_back(&n2); std::shared_ptr<E> e(new E); n1.edges.push_back(e); // don't add this twice if this is an edge from node_index1 back to itself if (node_index1 != node_index2) { n2.neighbors.push_back(&n1); n2.edges.push_back(e); } } catch (...) { clear(); throw; } } // ---------------------------------------------------------------------------------------- template < typename T, typename E, typename mem_manager, bool is_checked > void graph_kernel_1<T,E,mem_manager,is_checked>:: remove_edge ( unsigned long node_index1, unsigned long node_index2 ) { checker::check_remove_edge(node_index1, node_index2, *this); node_type& n1 = *nodes[node_index1]; node_type& n2 = *nodes[node_index2]; // remove the record of the link from n1 unsigned long pos = static_cast<unsigned long>(find(n1.neighbors.begin(), n1.neighbors.end(), &n2) - n1.neighbors.begin()); n1.neighbors.erase(n1.neighbors.begin() + pos); n1.edges.erase(n1.edges.begin() + pos); // check if this is an edge that goes from node_index1 back to itself if (node_index1 != node_index2) { // remove the record of the link from n2 unsigned long pos = static_cast<unsigned long>(find(n2.neighbors.begin(), n2.neighbors.end(), &n1) - n2.neighbors.begin()); n2.neighbors.erase(n2.neighbors.begin() + pos); n2.edges.erase(n2.edges.begin() + pos); } } // ---------------------------------------------------------------------------------------- template < typename T, typename E, typename mem_manager, bool is_checked > unsigned long graph_kernel_1<T,E,mem_manager,is_checked>:: add_node ( ) { try { std::shared_ptr<node_type> n(new node_type); n->idx = nodes.size(); nodes.push_back(n); return n->idx; } catch (...) { clear(); throw; } } // ---------------------------------------------------------------------------------------- template < typename T, typename E, typename mem_manager, bool is_checked > void graph_kernel_1<T,E,mem_manager,is_checked>:: remove_node ( unsigned long index ) { checker::check_remove_node(index,*this); node_type& n = *nodes[index]; // remove all edges pointing to this node from its neighbors for (unsigned long i = 0; i < n.neighbors.size(); ++i) { // remove the edge from this specific parent unsigned long pos = static_cast<unsigned long>(find(n.neighbors[i]->neighbors.begin(), n.neighbors[i]->neighbors.end(), &n) - n.neighbors[i]->neighbors.begin()); n.neighbors[i]->neighbors.erase(n.neighbors[i]->neighbors.begin() + pos); n.neighbors[i]->edges.erase(n.neighbors[i]->edges.begin() + pos); } // now remove this node by replacing it with the last node in the nodes vector nodes[index] = nodes[nodes.size()-1]; // update the index for the node we just moved nodes[index]->idx = index; // now remove the duplicated node at the end of the vector nodes.pop_back(); } // ---------------------------------------------------------------------------------------- } #endif // DLIB_GRAPH_KERNEl_1_