src/work/alpar/bfs.h
changeset 377 33fe0ee01dc5
equal deleted inserted replaced
-1:000000000000 0:7f81ab2917bd
       
     1 // -*-mode: c++; -*-
       
     2 #ifndef _BFS_H_
       
     3 #define _BFS_H_
       
     4 
       
     5 #include <queue>
       
     6 #include <graph.h>
       
     7 
       
     8 namespace NEGRO 
       
     9 {
       
    10   using namespace std;
       
    11   
       
    12   //   template <typename G,typename> class bfs_T
       
    13   //   {
       
    14   //     typedef G Graph;
       
    15   //     typedef Graph::OutEdgeIterator EdgeIterator; //Ez kell
       
    16   //     typedef Graph::NodeIterator NodeIterator;
       
    17     
       
    18   //     class bfs_node_D 
       
    19   //     {
       
    20   //       EdgeIterator Tree;
       
    21   //       int Dist;
       
    22   //       int Priority;
       
    23   //     }
       
    24   //   }
       
    25     
       
    26 
       
    27   //     //    template <bfs_T<G>>
       
    28   //     //    void bfs(bfs_T::Graph &G,const bfs_T::NodeIterator &start_node,const &maps)
       
    29 
       
    30     
       
    31   
       
    32   //   template <class G>
       
    33   //   class bfs_maps_T
       
    34   //   {
       
    35   //     typedef visited; //node->bool (RW)
       
    36   //     typedef tree;    //node->EdgeIterator (W)
       
    37   //     typedef dist;   //node->int (W)
       
    38   //     typedef priority; //node->int (W)
       
    39   //   };
       
    40   
       
    41   
       
    42   //   Nem jo! Osszeakad a masik Put-tal
       
    43   //   class do_nothing_map {};
       
    44   //   template <typename V,typename T>
       
    45   //   void Put(const do_nothing_map &p,const V &v,const T &t) {}
       
    46   
       
    47   struct do_nothing_map {
       
    48     template <typename V,typename T> static void Put(V v,T t) {}
       
    49     template <typename G> void SetG(G &g) {}
       
    50   };
       
    51   
       
    52 
       
    53   template <typename I,typename C, typename T, T C::*M>
       
    54   class class_element_map {
       
    55   public:
       
    56     typedef T value_type;
       
    57     static void Put(const I &i, const T &t) {(*i).*M=t;}
       
    58     static T Get(const I i) {return (*i).*M;}
       
    59     T &operator[](I i) {return (*i).*M;}
       
    60   };
       
    61 
       
    62   /* 
       
    63      template <typename C,typename I,typename T, T C::*M>
       
    64      void Put(class_element_map<C,T,M> p,I i, T t)
       
    65      {
       
    66      i->*M=t;   
       
    67      };
       
    68   */
       
    69 
       
    70   template <typename P,typename I,typename T>
       
    71   inline void Put(P &p,const I &i, const T &t)
       
    72   {
       
    73     p.Put(i,t);   
       
    74   };
       
    75   template <typename P,typename I>
       
    76   inline typename P::value_type Get(const P &p,const I &i)
       
    77   {
       
    78     return p.Get(i);   
       
    79   };
       
    80 
       
    81   /*
       
    82     template <typename G,typename T, typename G::EdgeType::*M>
       
    83     T Get(edge_element_map p,G::EdgeIterator i)
       
    84     {
       
    85     return i->*M;
       
    86     };
       
    87   */
       
    88   
       
    89   /* 
       
    90      template <class G>
       
    91      class default_bfs_maps
       
    92      {
       
    93      public:
       
    94      typedef typename G::NodeType NodeType;
       
    95     
       
    96      class_element_map<typename G::NodeIterator,
       
    97      typename G::NodeType,
       
    98      bool,&NodeType::isVis> visited; //node->bool (RW)
       
    99      do_nothing_map tree;   //node->EdgeIterator (W)
       
   100      do_nothing_map dist;   //node->int (W)
       
   101      do_nothing_map priority; //node->int (W)
       
   102      };
       
   103   */
       
   104 
       
   105   template<class G>
       
   106   struct bfs_node_data 
       
   107   {
       
   108     bool visited;
       
   109     typename G::EdgeIterator tree;
       
   110     int dist;
       
   111     int priority;
       
   112   };
       
   113   
       
   114   template <class G>
       
   115   class bfs_static_maps
       
   116   {
       
   117   public:
       
   118     typedef typename G::NodeType NodeType;
       
   119     
       
   120     /*    class_element_map<typename G::NodeIterator,
       
   121 	  typename G::NodeType,
       
   122 	  bfs_node_data<G>,&NT::D> n_d; //node-> data
       
   123     */
       
   124     class 
       
   125     {
       
   126     public:
       
   127       bfs_node_data<G> NodeType::*d;
       
   128       typedef bool value_type;
       
   129       void Put(typename G::NodeIterator &i, const value_type &t)
       
   130       {((*i).*d).visited=t;}
       
   131       value_type Get(const typename G::NodeIterator &i) const
       
   132       {return ((*i).*d).visited;}
       
   133     } visited;
       
   134     
       
   135     class 
       
   136     {
       
   137     public:
       
   138       bfs_node_data<G> NodeType::*d;
       
   139       typedef typename G::EdgeIterator value_type;
       
   140       void Put(typename G::NodeIterator &i, const value_type &t)
       
   141       {((*i).*d).tree=t;}
       
   142       value_type Get(const typename G::NodeIterator &i) const
       
   143       {return ((*i).*d).tree;}
       
   144     } tree;
       
   145     
       
   146     class 
       
   147     {
       
   148     public:
       
   149       bfs_node_data<G> NodeType::*d;
       
   150       typedef int value_type;
       
   151       void Put(typename G::NodeIterator &i, const value_type &t)
       
   152       {((*i).*d).dist=t;}
       
   153       value_type Get(const typename G::NodeIterator &i) const
       
   154       {return ((*i).*d).dist;}
       
   155     } dist;
       
   156     
       
   157     class 
       
   158     {
       
   159     public:
       
   160       bfs_node_data<G> NodeType::*d;
       
   161       typedef int value_type;
       
   162       void Put(typename G::NodeIterator &i,  const value_type &t)
       
   163       {((*i).*d).priority=t;}
       
   164       value_type Get(const typename G::NodeIterator &i) const
       
   165       {return ((*i).*d).priority;}
       
   166     } priority;
       
   167     
       
   168     //do_nothing_map tree;   //node->EdgeIterator (W)
       
   169     //    do_nothing_map dist;   //node->int (W)
       
   170     //    do_nothing_map priority; //node->int (W)
       
   171 
       
   172     void SetDataField(const bfs_node_data<G> NodeType::*dd)
       
   173     {
       
   174       tree.d=visited.d=dist.d=priority.d=dd;
       
   175     }
       
   176     
       
   177     bfs_static_maps(const bfs_node_data<G> NodeType::*dd) 
       
   178     {
       
   179       PutDataField(dd);
       
   180     }
       
   181     
       
   182     bfs_static_maps(const bfs_static_maps<G> &B) 
       
   183     {
       
   184       tree.d=B.tree.d;visited.d=B.visited.d;
       
   185       dist.d=B.dist.d;priority.d=B.priority.d;
       
   186     }
       
   187     
       
   188   };
       
   189   
       
   190   template<typename I>
       
   191   struct BFS_Q
       
   192   {
       
   193     I n;
       
   194     int dist;
       
   195     //  BFS_Q() {}
       
   196     //  BFS_Q(BFS_Q<I> &b) {n=b.n;dist=b.dist;}
       
   197   };
       
   198   
       
   199   template<class G,class M>
       
   200   void bfs_fn(G &Gr,const typename G::NodeIterator &start_node,M &maps)
       
   201   {
       
   202     using namespace std;
       
   203     
       
   204     typedef BFS_Q<typename G::NodeIterator> Q_T;
       
   205     
       
   206     Q_T q;
       
   207     
       
   208     int pr=0;
       
   209     typename G::NodeIterator n,m;
       
   210     typename G::OutEdgeIterator e;
       
   211     int d;
       
   212     
       
   213     for(Gr.GetFirst(n);n.isValid();++n)
       
   214       Put(maps.visited,n,false);
       
   215     
       
   216     queue<Q_T> Q;
       
   217     
       
   218     q.n=start_node;
       
   219     q.dist=0;
       
   220     Q.push(q);
       
   221     Put(maps.visited,start_node,true);
       
   222     //      Put(maps::tree,start_node,?????);
       
   223     Put(maps.dist,start_node,0);
       
   224     Put(maps.priority,start_node,pr++);
       
   225     
       
   226     do {
       
   227       n=Q.front().n;d=Q.front().dist+1;
       
   228       Q.pop();
       
   229       for(Gr.GetFirst(e,n);e.isValid();++e)
       
   230 	if(!Get(maps.visited,(m=e.Bnode()))) {
       
   231 	  q.n=m;
       
   232 	  q.dist=d;
       
   233 	  Q.push(q);
       
   234 	  Put(maps.visited,m,true);
       
   235 	  Put(maps.tree,m,e);
       
   236 	  Put(maps.dist,m,d);
       
   237 	  Put(maps.priority,m,pr++);
       
   238 	}
       
   239     } while(!Q.empty());
       
   240   };
       
   241 
       
   242   // bfs algorithm class
       
   243 
       
   244   template<class G>  //the default traits
       
   245   class default_bfs_T
       
   246   {
       
   247   public:
       
   248     
       
   249     typedef G Graph;
       
   250     typedef typename G::OutEdgeIterator SearchEdgeIterator;
       
   251     
       
   252     typedef typename G::NodeMap<bool> visited_map_t;
       
   253     typedef typename G::NodeMap<typename G::EdgeIterator> tree_map_t;
       
   254     
       
   255     typedef typename G::NodeMap<int> dist_map_t;   //node->int (W)
       
   256     typedef typename G::NodeMap<int> priority_map_t; //node->int (W)
       
   257 };
       
   258 
       
   259   template<class T>
       
   260   class Bfs
       
   261   {
       
   262   public:
       
   263     typedef typename T::Graph Graph;
       
   264     typedef typename Graph::NodeIterator NodeIterator;
       
   265     typedef typename Graph::EdgeIterator EdgeIterator;
       
   266     
       
   267     typedef typename T::SearchEdgeIterator SearchEdgeIterator;
       
   268     
       
   269     typename T::visited_map_t visited_map;
       
   270     typename T::tree_map_t tree_map;
       
   271     typename T::dist_map_t dist_map;
       
   272     typename T::priority_map_t priority_map;
       
   273     
       
   274     struct bfs_queue_cont
       
   275     {
       
   276       NodeIterator n;
       
   277       int dist;
       
   278     };
       
   279     
       
   280     std::queue<bfs_queue_cont> bfs_queue;
       
   281     
       
   282     int priority;
       
   283     Graph *G;
       
   284     
       
   285     //Bfs(int i): visited_map(G), tree_map(G), dist_map(G), priority_map(G) {}
       
   286     Bfs() {}
       
   287 
       
   288     void SetG(Graph &Gr)
       
   289     {
       
   290       G=&Gr;
       
   291       visited_map.SetG(Gr);
       
   292       tree_map.SetG(Gr);
       
   293       dist_map.SetG(Gr);
       
   294       priority_map.SetG(Gr);
       
   295     }
       
   296     
       
   297     void Init()
       
   298     {
       
   299       //There must be a better way to do this:
       
   300       while(!bfs_queue.empty()) bfs_queue.pop();
       
   301 
       
   302       for(NodeIterator n(*G);n.isValid();++n)
       
   303 	Put(visited_map,n,false);
       
   304       
       
   305       priority=0;
       
   306     }
       
   307     
       
   308     void AddStartNode(const NodeIterator &start_node,int dist=0)
       
   309     {
       
   310       bfs_queue_cont q;
       
   311       q.n=start_node;
       
   312       q.dist=dist;
       
   313       bfs_queue.push(q);
       
   314 
       
   315       Put(visited_map,start_node,true);
       
   316       //      Put(tree_map,start_node,?????);
       
   317       Put(dist_map,start_node,dist);
       
   318       Put(priority_map,start_node,priority++);    
       
   319     }
       
   320     
       
   321     void Init(const NodeIterator &start_node,int dist=0)
       
   322     {
       
   323       Init();
       
   324       AddStartNode(start_node,dist);
       
   325     }
       
   326 
       
   327     void Run() 
       
   328     {
       
   329       NodeIterator n,m;
       
   330       int d;
       
   331 
       
   332       bfs_queue_cont q;
       
   333       while(!(bfs_queue.empty()/* && other stop conditions */)) {
       
   334 	n=bfs_queue.front().n;d=bfs_queue.front().dist+1;
       
   335 	bfs_queue.pop();
       
   336 	for(SearchEdgeIterator e(*G,n);e.isValid();++e)
       
   337 	  if(!Get(visited_map,(m=e.Bnode()))) {
       
   338 	    q.n=m;
       
   339 	    q.dist=d;
       
   340 	    bfs_queue.push(q);
       
   341 	    Put(visited_map,m,true);
       
   342 	    Put(tree_map,m,e);
       
   343 	    Put(dist_map,m,d);
       
   344 	    Put(priority_map,m,priority++);
       
   345 	  }
       
   346       }
       
   347     }
       
   348   };
       
   349   
       
   350 }
       
   351   
       
   352 #endif