| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313 | 
//=======================================================================// Copyright 2008// Author: Matyas W Egyhazy//// Distributed under 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)//=======================================================================#ifndef BOOST_GRAPH_METRIC_TSP_APPROX_HPP#define BOOST_GRAPH_METRIC_TSP_APPROX_HPP// metric_tsp_approx// Generates an approximate tour solution for the traveling salesperson// problem in polynomial time. The current algorithm guarantees a tour with a// length at most as long as 2x optimal solution. The graph should have// 'natural' (metric) weights such that the triangle inequality is maintained.// Graphs must be fully interconnected.// TODO:// There are a couple of improvements that could be made.// 1) Change implementation to lower uppper bound Christofides heuristic// 2) Implement a less restrictive TSP heuristic (one that does not rely on//    triangle inequality).// 3) Determine if the algorithm can be implemented without creating a new//    graph.#include <vector>#include <boost/shared_ptr.hpp>#include <boost/concept_check.hpp>#include <boost/graph/graph_traits.hpp>#include <boost/graph/graph_as_tree.hpp>#include <boost/graph/adjacency_list.hpp>#include <boost/graph/prim_minimum_spanning_tree.hpp>#include <boost/graph/lookup_edge.hpp>#include <boost/throw_exception.hpp>namespace boost{    // Define a concept for the concept-checking library.    template <typename Visitor, typename Graph>    struct TSPVertexVisitorConcept    {    private:        Visitor vis_;    public:        typedef typename graph_traits<Graph>::vertex_descriptor Vertex;        BOOST_CONCEPT_USAGE(TSPVertexVisitorConcept)        {            Visitor vis(vis_);  // require copy construction            Graph g(1);            Vertex v(*vertices(g).first);            vis_.visit_vertex(v, g); // require visit_vertex        }    };    // Tree visitor that keeps track of a preorder traversal of a tree    // TODO: Consider migrating this to the graph_as_tree header.    // TODO: Parameterize the underlying stores o it doesn't have to be a vector.    template<typename Node, typename Tree> class PreorderTraverser    {    private:        std::vector<Node>& path_;    public:        typedef typename std::vector<Node>::const_iterator const_iterator;        PreorderTraverser(std::vector<Node>& p) : path_(p) {}        void preorder(Node n, const Tree&)        { path_.push_back(n); }        void inorder(Node, const Tree&) const {}        void postorder(Node, const Tree&) const {}        const_iterator begin() const { return path_.begin(); }        const_iterator end() const { return path_.end(); }    };    // Forward declarations    template <typename> class tsp_tour_visitor;    template <typename, typename, typename, typename> class tsp_tour_len_visitor;    template<typename VertexListGraph, typename OutputIterator>    void metric_tsp_approx_tour(VertexListGraph& g, OutputIterator o)    {        metric_tsp_approx_from_vertex(g, *vertices(g).first,            get(edge_weight, g), get(vertex_index, g),            tsp_tour_visitor<OutputIterator>(o));    }    template<typename VertexListGraph, typename WeightMap, typename OutputIterator>    void metric_tsp_approx_tour(VertexListGraph& g, WeightMap w, OutputIterator o)    {        metric_tsp_approx_from_vertex(g, *vertices(g).first,            w, tsp_tour_visitor<OutputIterator>(o));    }    template<typename VertexListGraph, typename OutputIterator>    void metric_tsp_approx_tour_from_vertex(VertexListGraph& g,        typename graph_traits<VertexListGraph>::vertex_descriptor start,        OutputIterator o)    {        metric_tsp_approx_from_vertex(g, start, get(edge_weight, g),            get(vertex_index, g), tsp_tour_visitor<OutputIterator>(o));    }    template<typename VertexListGraph, typename WeightMap,        typename OutputIterator>    void metric_tsp_approx_tour_from_vertex(VertexListGraph& g,    typename graph_traits<VertexListGraph>::vertex_descriptor start,        WeightMap w, OutputIterator o)    {        metric_tsp_approx_from_vertex(g, start, w, get(vertex_index, g),            tsp_tour_visitor<OutputIterator>(o));    }    template<typename VertexListGraph, typename TSPVertexVisitor>    void metric_tsp_approx(VertexListGraph& g, TSPVertexVisitor vis)    {        metric_tsp_approx_from_vertex(g, *vertices(g).first,            get(edge_weight, g), get(vertex_index, g), vis);    }    template<typename VertexListGraph, typename Weightmap,        typename VertexIndexMap, typename TSPVertexVisitor>    void metric_tsp_approx(VertexListGraph& g, Weightmap w,        TSPVertexVisitor vis)    {        metric_tsp_approx_from_vertex(g, *vertices(g).first, w,            get(vertex_index, g), vis);    }    template<typename VertexListGraph, typename WeightMap,        typename VertexIndexMap, typename TSPVertexVisitor>    void metric_tsp_approx(VertexListGraph& g, WeightMap w, VertexIndexMap id,        TSPVertexVisitor vis)    {        metric_tsp_approx_from_vertex(g, *vertices(g).first, w, id, vis);    }    template<typename VertexListGraph, typename WeightMap,        typename TSPVertexVisitor>    void metric_tsp_approx_from_vertex(VertexListGraph& g,    typename graph_traits<VertexListGraph>::vertex_descriptor start,        WeightMap w, TSPVertexVisitor vis)    {        metric_tsp_approx_from_vertex(g, start, w, get(vertex_index, g), vis);    }    template <        typename VertexListGraph,        typename WeightMap,        typename VertexIndexMap,        typename TSPVertexVisitor>    void metric_tsp_approx_from_vertex(const VertexListGraph& g,                                       typename graph_traits<VertexListGraph>::vertex_descriptor start,                                       WeightMap weightmap,                                       VertexIndexMap indexmap,                                       TSPVertexVisitor vis)    {        using namespace boost;        using namespace std;        BOOST_CONCEPT_ASSERT((VertexListGraphConcept<VertexListGraph>));        BOOST_CONCEPT_ASSERT((TSPVertexVisitorConcept<TSPVertexVisitor, VertexListGraph>));        // Types related to the input graph (GVertex is a template parameter).        typedef typename graph_traits<VertexListGraph>::vertex_descriptor GVertex;        typedef typename graph_traits<VertexListGraph>::vertex_iterator GVItr;        // We build a custom graph in this algorithm.        typedef adjacency_list <vecS, vecS, directedS, no_property, no_property > MSTImpl;        typedef graph_traits<MSTImpl>::vertex_descriptor Vertex;        typedef graph_traits<MSTImpl>::vertex_iterator VItr;        // And then re-cast it as a tree.        typedef iterator_property_map<vector<Vertex>::iterator, property_map<MSTImpl, vertex_index_t>::type> ParentMap;        typedef graph_as_tree<MSTImpl, ParentMap> Tree;        typedef tree_traits<Tree>::node_descriptor Node;        // A predecessor map.        typedef vector<GVertex> PredMap;        typedef iterator_property_map<typename PredMap::iterator, VertexIndexMap> PredPMap;        PredMap preds(num_vertices(g));        PredPMap pred_pmap(preds.begin(), indexmap);        // Compute a spanning tree over the in put g.        prim_minimum_spanning_tree(g, pred_pmap,             root_vertex(start)            .vertex_index_map(indexmap)            .weight_map(weightmap));        // Build a MST using the predecessor map from prim mst        MSTImpl mst(num_vertices(g));        std::size_t cnt = 0;        pair<VItr, VItr> mst_verts(vertices(mst));        for(typename PredMap::iterator vi(preds.begin()); vi != preds.end(); ++vi, ++cnt)        {            if(indexmap[*vi] != cnt) {                add_edge(*next(mst_verts.first, indexmap[*vi]),                         *next(mst_verts.first, cnt), mst);            }        }        // Build a tree abstraction over the MST.        vector<Vertex> parent(num_vertices(mst));        Tree t(mst, *vertices(mst).first,            make_iterator_property_map(parent.begin(),            get(vertex_index, mst)));        // Create tour using a preorder traversal of the mst        vector<Node> tour;        PreorderTraverser<Node, Tree> tvis(tour);        traverse_tree(indexmap[start], t, tvis);        pair<GVItr, GVItr> g_verts(vertices(g));        for(PreorderTraverser<Node, Tree>::const_iterator curr(tvis.begin());            curr != tvis.end(); ++curr)        {            // TODO: This is will be O(n^2) if vertex storage of g != vecS.            GVertex v = *next(g_verts.first, get(vertex_index, mst)[*curr]);            vis.visit_vertex(v, g);        }        // Connect back to the start of the tour        vis.visit_vertex(start, g);    }    // Default tsp tour visitor that puts the tour in an OutputIterator    template <typename OutItr>    class tsp_tour_visitor    {        OutItr itr_;    public:        tsp_tour_visitor(OutItr itr)            : itr_(itr)        { }        template <typename Vertex, typename Graph>        void visit_vertex(Vertex v, const Graph&)        {            BOOST_CONCEPT_ASSERT((OutputIterator<OutItr, Vertex>));            *itr_++ = v;        }    };    // Tsp tour visitor that adds the total tour length.    template<typename Graph, typename WeightMap, typename OutIter, typename Length>    class tsp_tour_len_visitor    {        typedef typename graph_traits<Graph>::vertex_descriptor Vertex;        BOOST_CONCEPT_ASSERT((OutputIterator<OutIter, Vertex>));        OutIter iter_;        Length& tourlen_;        WeightMap& wmap_;        Vertex previous_;        // Helper function for getting the null vertex.        Vertex null()        { return graph_traits<Graph>::null_vertex(); }    public:        tsp_tour_len_visitor(Graph const&, OutIter iter, Length& l, WeightMap map)            : iter_(iter), tourlen_(l), wmap_(map), previous_(null())        { }        void visit_vertex(Vertex v, const Graph& g)        {            typedef typename graph_traits<Graph>::edge_descriptor Edge;            // If it is not the start, then there is a            // previous vertex            if(previous_ != null())            {                // NOTE: For non-adjacency matrix graphs g, this bit of code                // will be linear in the degree of previous_ or v. A better                // solution would be to visit edges of the graph, but that                // would require revisiting the core algorithm.                Edge e;                bool found;                boost::tie(e, found) = lookup_edge(previous_, v, g);                if(!found) {                    BOOST_THROW_EXCEPTION(not_complete());                }                tourlen_ += wmap_[e];            }            previous_ = v;            *iter_++ = v;        }    };    // Object generator(s)    template <typename OutIter>    inline tsp_tour_visitor<OutIter>    make_tsp_tour_visitor(OutIter iter)    { return tsp_tour_visitor<OutIter>(iter); }    template <typename Graph, typename WeightMap, typename OutIter, typename Length>    inline tsp_tour_len_visitor<Graph, WeightMap, OutIter, Length>    make_tsp_tour_len_visitor(Graph const& g, OutIter iter, Length& l, WeightMap map)    { return tsp_tour_len_visitor<Graph, WeightMap, OutIter, Length>(g, iter, l, map); }} //boost#endif // BOOST_GRAPH_METRIC_TSP_APPROX_HPP
 |