201 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
			
		
		
	
	
			201 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
/* ----------------------------------------------------------------------------
 | 
						|
 | 
						|
 * GTSAM Copyright 2010, Georgia Tech Research Corporation, 
 | 
						|
 * Atlanta, Georgia 30332-0415
 | 
						|
 * All Rights Reserved
 | 
						|
 * Authors: Frank Dellaert, et al. (see THANKS for the full author list)
 | 
						|
 | 
						|
 * See LICENSE for the license information
 | 
						|
 | 
						|
 * -------------------------------------------------------------------------- */
 | 
						|
 | 
						|
/**
 | 
						|
 * @file    FactorGraph.h
 | 
						|
 * @brief   Factor Graph Base Class
 | 
						|
 * @author  Carlos Nieto
 | 
						|
 * @author  Christian Potthast
 | 
						|
 * @author  Michael Kaess
 | 
						|
 */
 | 
						|
 | 
						|
// \callgraph
 | 
						|
 | 
						|
#pragma once
 | 
						|
 | 
						|
#include <boost/shared_ptr.hpp>
 | 
						|
#include <boost/foreach.hpp>
 | 
						|
#include <boost/serialization/nvp.hpp>
 | 
						|
#include <boost/pool/pool_alloc.hpp>
 | 
						|
 | 
						|
#include <gtsam/base/types.h>
 | 
						|
#include <gtsam/base/Testable.h>
 | 
						|
#include <gtsam/inference/BayesNet.h>
 | 
						|
#include <gtsam/inference/graph.h>
 | 
						|
 | 
						|
namespace gtsam {
 | 
						|
 | 
						|
	/**
 | 
						|
	 * A factor graph is a bipartite graph with factor nodes connected to variable nodes.
 | 
						|
	 * In this class, however, only factor nodes are kept around.
 | 
						|
	 * 
 | 
						|
	 * Templated on the type of factors and values structure.
 | 
						|
	 */
 | 
						|
	template<class FACTOR>
 | 
						|
	class FactorGraph: public Testable<FactorGraph<FACTOR> > {
 | 
						|
	public:
 | 
						|
	  typedef FACTOR Factor;
 | 
						|
	  typedef boost::shared_ptr<FactorGraph<FACTOR> > shared_ptr;
 | 
						|
		typedef typename boost::shared_ptr<FACTOR> sharedFactor;
 | 
						|
		typedef typename std::vector<sharedFactor>::iterator iterator;
 | 
						|
		typedef typename std::vector<sharedFactor>::const_iterator const_iterator;
 | 
						|
 | 
						|
	protected:
 | 
						|
 | 
						|
    /** Collection of factors */
 | 
						|
		std::vector<sharedFactor> factors_;
 | 
						|
 | 
						|
	public:
 | 
						|
 | 
						|
		/** ------------------ Creating Factor Graphs ---------------------------- */
 | 
						|
 | 
						|
		/** Default constructor */
 | 
						|
		FactorGraph() {}
 | 
						|
 | 
						|
		/** convert from Bayes net */
 | 
						|
		template<class CONDITIONAL>
 | 
						|
		FactorGraph(const BayesNet<CONDITIONAL>& bayesNet);
 | 
						|
 | 
						|
		/** convert from a derived type */
 | 
						|
		template<class DERIVEDFACTOR>
 | 
						|
		FactorGraph(const FactorGraph<DERIVEDFACTOR>& factorGraph);
 | 
						|
 | 
						|
		/** Add a factor */
 | 
						|
		template<class DERIVEDFACTOR>
 | 
						|
		void push_back(const boost::shared_ptr<DERIVEDFACTOR>& factor);
 | 
						|
 | 
						|
		/** push back many factors */
 | 
						|
		void push_back(const FactorGraph<FACTOR>& factors);
 | 
						|
 | 
						|
		/** push back many factors with an iterator */
 | 
						|
		template<typename ITERATOR>
 | 
						|
		void push_back(ITERATOR firstFactor, ITERATOR lastFactor);
 | 
						|
 | 
						|
		/** ------------------ Querying Factor Graphs ---------------------------- */
 | 
						|
 | 
						|
		/** print out graph */
 | 
						|
		void print(const std::string& s = "FactorGraph") const;
 | 
						|
 | 
						|
		/** Check equality */
 | 
						|
		bool equals(const FactorGraph& fg, double tol = 1e-9) const;
 | 
						|
 | 
						|
		/** const cast to the underlying vector of factors */
 | 
						|
		operator const std::vector<sharedFactor>&() const { return factors_; }
 | 
						|
 | 
						|
		/** STL begin and end, so we can use BOOST_FOREACH */
 | 
						|
		const_iterator begin() const { return factors_.begin();}
 | 
						|
		const_iterator end()   const { return factors_.end();  }
 | 
						|
 | 
						|
		/** Get a specific factor by index */
 | 
						|
		sharedFactor operator[](size_t i) const { assert(i<factors_.size()); return factors_[i]; }
 | 
						|
 | 
						|
    /** Get the first factor */
 | 
						|
    sharedFactor front() const { return factors_.front(); }
 | 
						|
 | 
						|
		/** Get the last factor */
 | 
						|
		sharedFactor back() const { return factors_.back(); }
 | 
						|
 | 
						|
		/** return the number of factors and NULLS */
 | 
						|
		size_t size() const { return factors_.size();}
 | 
						|
 | 
						|
		/** return the number valid factors */
 | 
						|
		size_t nrFactors() const;
 | 
						|
 | 
						|
		/** ----------------- Modifying Factor Graphs ---------------------------- */
 | 
						|
 | 
						|
		/** STL begin and end, so we can use BOOST_FOREACH */
 | 
						|
		iterator begin()       { return factors_.begin();}
 | 
						|
		iterator end()         { return factors_.end();  }
 | 
						|
 | 
						|
		/**
 | 
						|
		 * Reserve space for the specified number of factors if you know in
 | 
						|
		 * advance how many there will be (works like std::vector::reserve).
 | 
						|
		 */
 | 
						|
		void reserve(size_t size) { factors_.reserve(size); }
 | 
						|
 | 
						|
		/** delete factor without re-arranging indexes by inserting a NULL pointer */
 | 
						|
		inline void remove(size_t i) { factors_[i].reset();}
 | 
						|
 | 
						|
		/** replace a factor by index */
 | 
						|
		void replace(size_t index, sharedFactor factor);
 | 
						|
 | 
						|
	private:
 | 
						|
 | 
						|
		/** Serialization function */
 | 
						|
		friend class boost::serialization::access;
 | 
						|
		template<class ARCHIVE>
 | 
						|
		void serialize(ARCHIVE & ar, const unsigned int version) {
 | 
						|
			ar & BOOST_SERIALIZATION_NVP(factors_);
 | 
						|
		}
 | 
						|
	}; // FactorGraph
 | 
						|
 | 
						|
	/**
 | 
						|
   * static function that combines two factor graphs
 | 
						|
   * @param const &fg1 Linear factor graph
 | 
						|
   * @param const &fg2 Linear factor graph
 | 
						|
   * @return a new combined factor graph
 | 
						|
   */
 | 
						|
	template<class FACTORGRAPH>
 | 
						|
	FACTORGRAPH combine(const FACTORGRAPH& fg1, const FACTORGRAPH& fg2);
 | 
						|
 | 
						|
	/**
 | 
						|
	 * These functions are defined here because they are templated on an
 | 
						|
	 * additional parameter.  Putting them in the -inl.h file would mean these
 | 
						|
	 * would have to be explicitly instantiated for any possible derived factor
 | 
						|
	 * type.
 | 
						|
	 */
 | 
						|
 | 
						|
  /* ************************************************************************* */
 | 
						|
  template<class FACTOR>
 | 
						|
  template<class DERIVEDFACTOR>
 | 
						|
  FactorGraph<FACTOR>::FactorGraph(const FactorGraph<DERIVEDFACTOR>& factorGraph) {
 | 
						|
    factors_.reserve(factorGraph.size());
 | 
						|
    BOOST_FOREACH(const typename DERIVEDFACTOR::shared_ptr& factor, factorGraph) {
 | 
						|
      if(factor)
 | 
						|
        this->push_back(sharedFactor(new FACTOR(*factor)));
 | 
						|
      else
 | 
						|
        this->push_back(sharedFactor());
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /* ************************************************************************* */
 | 
						|
  template<class FACTOR>
 | 
						|
  template<class CONDITIONAL>
 | 
						|
  FactorGraph<FACTOR>::FactorGraph(const BayesNet<CONDITIONAL>& bayesNet) {
 | 
						|
    factors_.reserve(bayesNet.size());
 | 
						|
    BOOST_FOREACH(const typename CONDITIONAL::shared_ptr& cond, bayesNet) {
 | 
						|
      this->push_back(sharedFactor(new FACTOR(*cond)));
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /* ************************************************************************* */
 | 
						|
  template<class FACTOR>
 | 
						|
  template<class DERIVEDFACTOR>
 | 
						|
  inline void FactorGraph<FACTOR>::push_back(const boost::shared_ptr<DERIVEDFACTOR>& factor) {
 | 
						|
#ifndef NDEBUG
 | 
						|
    factors_.push_back(boost::dynamic_pointer_cast<FACTOR>(factor)); // add the actual factor
 | 
						|
#else
 | 
						|
    factors_.push_back(boost::static_pointer_cast<FACTOR>(factor));
 | 
						|
#endif
 | 
						|
  }
 | 
						|
 | 
						|
  /* ************************************************************************* */
 | 
						|
  template<class FACTOR>
 | 
						|
  template<typename ITERATOR>
 | 
						|
  void FactorGraph<FACTOR>::push_back(ITERATOR firstFactor, ITERATOR lastFactor) {
 | 
						|
    ITERATOR factor = firstFactor;
 | 
						|
    while(factor != lastFactor)
 | 
						|
      this->push_back(*(factor++));
 | 
						|
  }
 | 
						|
 | 
						|
} // namespace gtsam
 | 
						|
 |