| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520 | // Copyright (C) 2007 Douglas Gregor// Use, modification and distribution is subject to the Boost Software// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at// http://www.boost.org/LICENSE_1_0.txt)// Provides support for named vertices in graphs, allowing one to more// easily associate unique external names (URLs, city names, employee// ID numbers, etc.) with the vertices of a graph.#ifndef BOOST_GRAPH_NAMED_GRAPH_HPP#define BOOST_GRAPH_NAMED_GRAPH_HPP#include <boost/config.hpp>#include <boost/static_assert.hpp>#include <boost/functional/hash.hpp>#include <boost/graph/graph_traits.hpp>#include <boost/graph/properties.hpp>#include <boost/multi_index/hashed_index.hpp>#include <boost/multi_index/member.hpp>#include <boost/multi_index_container.hpp>#include <boost/optional.hpp>#include <boost/pending/property.hpp> // for boost::lookup_one_property#include <boost/pending/container_traits.hpp>#include <boost/throw_exception.hpp>#include <boost/tuple/tuple.hpp> // for boost::make_tuple#include <boost/type_traits/is_same.hpp>#include <boost/type_traits/is_base_of.hpp>#include <boost/type_traits/remove_cv.hpp>#include <boost/type_traits/remove_reference.hpp>#include <boost/utility/enable_if.hpp>#include <functional> // for std::equal_to#include <stdexcept> // for std::runtime_error#include <utility> // for std::pairnamespace boost{namespace graph{    /*******************************************************************     * User-customized traits                                          *     *******************************************************************/    /**     * @brief Trait used to extract the internal vertex name from a vertex     * property.     *     * To enable the use of internal vertex names in a graph type,     * specialize the @c internal_vertex_name trait for your graph     * property (e.g., @c a City class, which stores information about the     * vertices in a road map).     */    template < typename VertexProperty > struct internal_vertex_name    {        /**         *  The @c type field provides a function object that extracts a key         *  from the @c VertexProperty. The function object type must have a         *  nested @c result_type that provides the type of the key. For         *  more information, see the @c KeyExtractor concept in the         *  Boost.MultiIndex documentation: @c type must either be @c void         *  (if @c VertexProperty does not have an internal vertex name) or         *  a model of @c KeyExtractor.         */        typedef void type;    };    /**     * Extract the internal vertex name from a @c property structure by     * looking at its base.     */    template < typename Tag, typename T, typename Base >    struct internal_vertex_name< property< Tag, T, Base > >    : internal_vertex_name< Base >    {    };    /**     * Construct an instance of @c VertexProperty directly from its     * name. This function object should be used within the @c     * internal_vertex_constructor trait.     */    template < typename VertexProperty > struct vertex_from_name    {    private:        typedef typename internal_vertex_name< VertexProperty >::type            extract_name_type;        typedef typename remove_cv< typename remove_reference<            typename extract_name_type::result_type >::type >::type            vertex_name_type;    public:        typedef vertex_name_type argument_type;        typedef VertexProperty result_type;        VertexProperty operator()(const vertex_name_type& name)        {            return VertexProperty(name);        }    };    /**     * Throw an exception whenever one tries to construct a @c     * VertexProperty instance from its name.     */    template < typename VertexProperty > struct cannot_add_vertex    {    private:        typedef typename internal_vertex_name< VertexProperty >::type            extract_name_type;        typedef typename remove_cv< typename remove_reference<            typename extract_name_type::result_type >::type >::type            vertex_name_type;    public:        typedef vertex_name_type argument_type;        typedef VertexProperty result_type;        VertexProperty operator()(const vertex_name_type&)        {            boost::throw_exception(                std::runtime_error("add_vertex: "                                   "unable to create a vertex from its name"));        }    };    /**     * @brief Trait used to construct an instance of a @c VertexProperty,     * which is a class type that stores the properties associated with a     * vertex in a graph, from just the name of that vertex property. This     * operation is used when an operation is required to map from a     * vertex name to a vertex descriptor (e.g., to add an edge outgoing     * from that vertex), but no vertex by the name exists. The function     * object provided by this trait will be used to add new vertices     * based only on their names. Since this cannot be done for arbitrary     * types, the default behavior is to throw an exception when this     * routine is called, which requires that all named vertices be added     * before adding any edges by name.     */    template < typename VertexProperty > struct internal_vertex_constructor    {        /**         * The @c type field provides a function object that constructs a         * new instance of @c VertexProperty from the name of the vertex (as         * determined by @c internal_vertex_name). The function object shall         * accept a vertex name and return a @c VertexProperty. Predefined         * options include:         *         *   @c vertex_from_name<VertexProperty>: construct an instance of         *   @c VertexProperty directly from the name.         *         *   @c cannot_add_vertex<VertexProperty>: the default value, which         *   throws an @c std::runtime_error if one attempts to add a vertex         *   given just the name.         */        typedef cannot_add_vertex< VertexProperty > type;    };    /**     * Extract the internal vertex constructor from a @c property structure by     * looking at its base.     */    template < typename Tag, typename T, typename Base >    struct internal_vertex_constructor< property< Tag, T, Base > >    : internal_vertex_constructor< Base >    {    };    /*******************************************************************     * Named graph mixin                                               *     *******************************************************************/    /**     * named_graph is a mixin that provides names for the vertices of a     * graph, including a mapping from names to vertices. Graph types that     * may or may not be have vertex names (depending on the properties     * supplied by the user) should use maybe_named_graph.     *     * Template parameters:     *     *   Graph: the graph type that derives from named_graph     *     *   Vertex: the type of a vertex descriptor in Graph. Note: we cannot     *   use graph_traits here, because the Graph is not yet defined.     *     *   VertexProperty: the type stored with each vertex in the Graph.     */    template < typename Graph, typename Vertex, typename VertexProperty >    class named_graph    {    public:        /// The type of the function object that extracts names from vertex        /// properties.        typedef typename internal_vertex_name< VertexProperty >::type            extract_name_type;        /// The type of the "bundled" property, from which the name can be        /// extracted.        typedef typename lookup_one_property< VertexProperty,            vertex_bundle_t >::type bundled_vertex_property_type;        /// The type of the function object that generates vertex properties        /// from names, for the implicit addition of vertices.        typedef typename internal_vertex_constructor< VertexProperty >::type            vertex_constructor_type;        /// The type used to name vertices in the graph        typedef typename remove_cv< typename remove_reference<            typename extract_name_type::result_type >::type >::type            vertex_name_type;        /// The type of vertex descriptors in the graph        typedef Vertex vertex_descriptor;    private:        /// Key extractor for use with the multi_index_container        struct extract_name_from_vertex        {            typedef vertex_name_type result_type;            extract_name_from_vertex(                Graph& graph, const extract_name_type& extract)            : graph(graph), extract(extract)            {            }            const result_type& operator()(Vertex vertex) const            {                return extract(graph[vertex]);            }            Graph& graph;            extract_name_type extract;        };    public:        /// The type that maps names to vertices        typedef multi_index::multi_index_container< Vertex,            multi_index::indexed_by<                multi_index::hashed_unique< multi_index::tag< vertex_name_t >,                    extract_name_from_vertex > > >            named_vertices_type;        /// The set of vertices, indexed by name        typedef            typename named_vertices_type::template index< vertex_name_t >::type                vertices_by_name_type;        /// Construct an instance of the named graph mixin, using the given        /// function object to extract a name from the bundled property        /// associated with a vertex.        named_graph(const extract_name_type& extract = extract_name_type(),            const vertex_constructor_type& vertex_constructor            = vertex_constructor_type());        /// Notify the named_graph that we have added the given vertex. The        /// name of the vertex will be added to the mapping.        void added_vertex(Vertex vertex);        /// Notify the named_graph that we are removing the given        /// vertex. The name of the vertex will be removed from the mapping.        template < typename VertexIterStability >        void removing_vertex(Vertex vertex, VertexIterStability);        /// Notify the named_graph that we are clearing the graph.        /// This will clear out all of the name->vertex mappings        void clearing_graph();        /// Retrieve the derived instance        Graph& derived() { return static_cast< Graph& >(*this); }        const Graph& derived() const        {            return static_cast< const Graph& >(*this);        }        /// Extract the name from a vertex property instance        typename extract_name_type::result_type extract_name(            const bundled_vertex_property_type& property);        /// Search for a vertex that has the given property (based on its        /// name)        optional< vertex_descriptor > vertex_by_property(            const bundled_vertex_property_type& property);        /// Mapping from names to vertices        named_vertices_type named_vertices;        /// Constructs a vertex from the name of that vertex        vertex_constructor_type vertex_constructor;    };/// Helper macro containing the template parameters of named_graph#define BGL_NAMED_GRAPH_PARAMS \    typename Graph, typename Vertex, typename VertexProperty/// Helper macro containing the named_graph<...> instantiation#define BGL_NAMED_GRAPH named_graph< Graph, Vertex, VertexProperty >    template < BGL_NAMED_GRAPH_PARAMS >    BGL_NAMED_GRAPH::named_graph(const extract_name_type& extract,        const vertex_constructor_type& vertex_constructor)    : named_vertices(typename named_vertices_type::ctor_args_list(        boost::make_tuple(boost::make_tuple(0, // initial number of buckets            extract_name_from_vertex(derived(), extract),            boost::hash< vertex_name_type >(),            std::equal_to< vertex_name_type >()))))    , vertex_constructor(vertex_constructor)    {    }    template < BGL_NAMED_GRAPH_PARAMS >    inline void BGL_NAMED_GRAPH::added_vertex(Vertex vertex)    {        named_vertices.insert(vertex);    }    template < BGL_NAMED_GRAPH_PARAMS >    template < typename VertexIterStability >    inline void BGL_NAMED_GRAPH::removing_vertex(        Vertex vertex, VertexIterStability)    {        BOOST_STATIC_ASSERT_MSG(            (boost::is_base_of< boost::graph_detail::stable_tag,                VertexIterStability >::value),            "Named graphs cannot use vecS as vertex container and remove "            "vertices; the lack of vertex descriptor stability (which iterator "            "stability is a proxy for) means that the name -> vertex mapping "            "would need to be completely rebuilt after each deletion.  See "            "https://svn.boost.org/trac/boost/ticket/7863 for more information "            "and a test case.");        typedef typename BGL_NAMED_GRAPH::vertex_name_type vertex_name_type;        const vertex_name_type& vertex_name = extract_name(derived()[vertex]);        named_vertices.erase(vertex_name);    }    template < BGL_NAMED_GRAPH_PARAMS >    inline void BGL_NAMED_GRAPH::clearing_graph()    {        named_vertices.clear();    }    template < BGL_NAMED_GRAPH_PARAMS >    typename BGL_NAMED_GRAPH::extract_name_type::result_type    BGL_NAMED_GRAPH::extract_name(const bundled_vertex_property_type& property)    {        return named_vertices.key_extractor().extract(property);    }    template < BGL_NAMED_GRAPH_PARAMS >    optional< typename BGL_NAMED_GRAPH::vertex_descriptor >    BGL_NAMED_GRAPH::vertex_by_property(        const bundled_vertex_property_type& property)    {        return find_vertex(extract_name(property), *this);    }    /// Retrieve the vertex associated with the given name    template < BGL_NAMED_GRAPH_PARAMS >    optional< Vertex > find_vertex(        typename BGL_NAMED_GRAPH::vertex_name_type const& name,        const BGL_NAMED_GRAPH& g)    {        typedef typename BGL_NAMED_GRAPH::vertices_by_name_type            vertices_by_name_type;        // Retrieve the set of vertices indexed by name        vertices_by_name_type const& vertices_by_name            = g.named_vertices.template get< vertex_name_t >();        /// Look for a vertex with the given name        typename vertices_by_name_type::const_iterator iter            = vertices_by_name.find(name);        if (iter == vertices_by_name.end())            return optional< Vertex >(); // vertex not found        else            return *iter;    }    /// Retrieve the vertex associated with the given name, or add a new    /// vertex with that name if no such vertex is available.    /// Note: This is enabled only when the vertex property type is different    ///       from the vertex name to avoid ambiguous overload problems with    ///       the add_vertex() function that takes a vertex property.    template < BGL_NAMED_GRAPH_PARAMS >    typename disable_if<        is_same< typename BGL_NAMED_GRAPH::vertex_name_type, VertexProperty >,        Vertex >::type    add_vertex(typename BGL_NAMED_GRAPH::vertex_name_type const& name,        BGL_NAMED_GRAPH& g)    {        if (optional< Vertex > vertex = find_vertex(name, g))            /// We found the vertex, so return it            return *vertex;        else            /// There is no vertex with the given name, so create one            return add_vertex(g.vertex_constructor(name), g.derived());    }    /// Add an edge using vertex names to refer to the vertices    template < BGL_NAMED_GRAPH_PARAMS >    std::pair< typename graph_traits< Graph >::edge_descriptor, bool > add_edge(        typename BGL_NAMED_GRAPH::vertex_name_type const& u_name,        typename BGL_NAMED_GRAPH::vertex_name_type const& v_name,        BGL_NAMED_GRAPH& g)    {        return add_edge(add_vertex(u_name, g.derived()),            add_vertex(v_name, g.derived()), g.derived());    }    /// Add an edge using vertex descriptors or names to refer to the vertices    template < BGL_NAMED_GRAPH_PARAMS >    std::pair< typename graph_traits< Graph >::edge_descriptor, bool > add_edge(        typename BGL_NAMED_GRAPH::vertex_descriptor const& u,        typename BGL_NAMED_GRAPH::vertex_name_type const& v_name,        BGL_NAMED_GRAPH& g)    {        return add_edge(u, add_vertex(v_name, g.derived()), g.derived());    }    /// Add an edge using vertex descriptors or names to refer to the vertices    template < BGL_NAMED_GRAPH_PARAMS >    std::pair< typename graph_traits< Graph >::edge_descriptor, bool > add_edge(        typename BGL_NAMED_GRAPH::vertex_name_type const& u_name,        typename BGL_NAMED_GRAPH::vertex_descriptor const& v,        BGL_NAMED_GRAPH& g)    {        return add_edge(add_vertex(u_name, g.derived()), v, g.derived());    }    // Overloads to support EdgeMutablePropertyGraph graphs    template < BGL_NAMED_GRAPH_PARAMS >    std::pair< typename graph_traits< Graph >::edge_descriptor, bool > add_edge(        typename BGL_NAMED_GRAPH::vertex_descriptor const& u,        typename BGL_NAMED_GRAPH::vertex_name_type const& v_name,        typename edge_property_type< Graph >::type const& p, BGL_NAMED_GRAPH& g)    {        return add_edge(u, add_vertex(v_name, g.derived()), p, g.derived());    }    template < BGL_NAMED_GRAPH_PARAMS >    std::pair< typename graph_traits< Graph >::edge_descriptor, bool > add_edge(        typename BGL_NAMED_GRAPH::vertex_name_type const& u_name,        typename BGL_NAMED_GRAPH::vertex_descriptor const& v,        typename edge_property_type< Graph >::type const& p, BGL_NAMED_GRAPH& g)    {        return add_edge(add_vertex(u_name, g.derived()), v, p, g.derived());    }    template < BGL_NAMED_GRAPH_PARAMS >    std::pair< typename graph_traits< Graph >::edge_descriptor, bool > add_edge(        typename BGL_NAMED_GRAPH::vertex_name_type const& u_name,        typename BGL_NAMED_GRAPH::vertex_name_type const& v_name,        typename edge_property_type< Graph >::type const& p, BGL_NAMED_GRAPH& g)    {        return add_edge(add_vertex(u_name, g.derived()),            add_vertex(v_name, g.derived()), p, g.derived());    }#undef BGL_NAMED_GRAPH#undef BGL_NAMED_GRAPH_PARAMS    /*******************************************************************     * Maybe named graph mixin                                         *     *******************************************************************/    /**     * A graph mixin that can provide a mapping from names to vertices,     * and use that mapping to simplify creation and manipulation of     * graphs.     */    template < typename Graph, typename Vertex, typename VertexProperty,        typename ExtractName =            typename internal_vertex_name< VertexProperty >::type >    struct maybe_named_graph    : public named_graph< Graph, Vertex, VertexProperty >    {    };    /**     * A graph mixin that can provide a mapping from names to vertices,     * and use that mapping to simplify creation and manipulation of     * graphs. This partial specialization turns off this functionality     * when the @c VertexProperty does not have an internal vertex name.     */    template < typename Graph, typename Vertex, typename VertexProperty >    struct maybe_named_graph< Graph, Vertex, VertexProperty, void >    {        /// The type of the "bundled" property, from which the name can be        /// extracted.        typedef typename lookup_one_property< VertexProperty,            vertex_bundle_t >::type bundled_vertex_property_type;        /// Notify the named_graph that we have added the given vertex. This        /// is a no-op.        void added_vertex(Vertex) {}        /// Notify the named_graph that we are removing the given        /// vertex. This is a no-op.        template < typename VertexIterStability >        void removing_vertex(Vertex, VertexIterStability)        {        }        /// Notify the named_graph that we are clearing the graph. This is a        /// no-op.        void clearing_graph() {}        /// Search for a vertex that has the given property (based on its        /// name). This always returns an empty optional<>        optional< Vertex > vertex_by_property(            const bundled_vertex_property_type&)        {            return optional< Vertex >();        }    };}} // end namespace boost::graph#endif // BOOST_GRAPH_NAMED_GRAPH_HPP
 |