| 
									
										
										
										
											2010-10-14 12:54:38 +08:00
										 |  |  | /* ----------------------------------------------------------------------------
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  * 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 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  * -------------------------------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * @file    Factor.h | 
					
						
							|  |  |  |  * @brief   A simple factor class to use in a factor graph | 
					
						
							|  |  |  |  * @brief   factor | 
					
						
							|  |  |  |  * @author  Kai Ni | 
					
						
							| 
									
										
										
										
											2009-10-23 01:23:24 +08:00
										 |  |  |  * @author  Frank Dellaert | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // \callgraph
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | #include <vector>
 | 
					
						
							|  |  |  | #include <map>
 | 
					
						
							| 
									
										
										
										
											2009-10-07 02:25:04 +08:00
										 |  |  | #include <boost/utility.hpp> // for noncopyable
 | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | #include <boost/serialization/nvp.hpp>
 | 
					
						
							|  |  |  | #include <gtsam/base/types.h>
 | 
					
						
							| 
									
										
										
										
											2010-08-20 01:23:19 +08:00
										 |  |  | #include <gtsam/base/Testable.h>
 | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  | #include <gtsam/base/FastMap.h>
 | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | #include <gtsam/inference/inference.h>
 | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace gtsam { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  | template<class KEY> class ConditionalBase; | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * A simple factor class to use in a factor graph. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * We make it noncopyable so we enforce the fact that factors are | 
					
						
							|  |  |  |  * kept in pointer containers. To be safe, you should make them | 
					
						
							|  |  |  |  * immutable, i.e., practicing functional programming. However, this | 
					
						
							|  |  |  |  * conflicts with efficiency as well, esp. in the case of incomplete | 
					
						
							|  |  |  |  * QR factorization. A solution is still being sought. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-10-09 11:09:58 +08:00
										 |  |  |  * A Factor is templated on a Values, for example VectorValues is a values structure of | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  |  * labeled vectors. This way, we can have factors that might be defined on discrete | 
					
						
							|  |  |  |  * variables, continuous ones, or a combination of both. It is up to the config to | 
					
						
							|  |  |  |  * provide the appropriate values at the appropriate time. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  | template<typename KEY> | 
					
						
							|  |  |  | class FactorBase : public Testable<FactorBase<KEY> > { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   typedef KEY Key; | 
					
						
							|  |  |  |   typedef FactorBase<Key> This; | 
					
						
							|  |  |  |   typedef gtsam::ConditionalBase<Key> Conditional; | 
					
						
							|  |  |  |   typedef boost::shared_ptr<FactorBase> shared_ptr; | 
					
						
							|  |  |  |   typedef std::vector<Index>::iterator iterator; | 
					
						
							|  |  |  |   typedef std::vector<Index>::const_iterator const_iterator; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | protected: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   std::vector<Key> keys_; | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-14 04:43:58 +08:00
										 |  |  |   /** Internal check to make sure keys are sorted.
 | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  |    * If NDEBUG is defined, this is empty and optimized out. */ | 
					
						
							| 
									
										
										
										
											2010-10-14 04:43:58 +08:00
										 |  |  |   void assertInvariants() const; | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** Copy constructor */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   FactorBase(const This& f); | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Construct from derived type */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   FactorBase(const Conditional& c); | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Constructor from a collection of keys */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   template<class KeyIterator> FactorBase(KeyIterator beginKey, KeyIterator endKey) : | 
					
						
							|  |  |  |         keys_(beginKey, endKey) { assertInvariants(); } | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Default constructor for I/O */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   FactorBase() {} | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Construct unary factor */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   FactorBase(Key key) : keys_(1) { | 
					
						
							| 
									
										
										
										
											2010-10-14 04:43:58 +08:00
										 |  |  |     keys_[0] = key; assertInvariants(); } | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Construct binary factor */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   FactorBase(Key key1, Key key2) : keys_(2) { | 
					
						
							| 
									
										
										
										
											2010-10-14 04:43:58 +08:00
										 |  |  |     keys_[0] = key1; keys_[1] = key2; assertInvariants(); } | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Construct ternary factor */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   FactorBase(Key key1, Key key2, Key key3) : keys_(3) { | 
					
						
							| 
									
										
										
										
											2010-10-14 04:43:58 +08:00
										 |  |  |     keys_[0] = key1; keys_[1] = key2; keys_[2] = key3; assertInvariants(); } | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Construct 4-way factor */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   FactorBase(Key key1, Key key2, Key key3, Key key4) : keys_(4) { | 
					
						
							| 
									
										
										
										
											2010-10-14 04:43:58 +08:00
										 |  |  |     keys_[0] = key1; keys_[1] = key2; keys_[2] = key3; keys_[3] = key4; assertInvariants(); } | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Named constructor for combining a set of factors with pre-computed set of
 | 
					
						
							|  |  |  |    * variables.  (Old style - will be removed when scalar elimination is | 
					
						
							|  |  |  |    * removed in favor of the EliminationTree). */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   template<class DERIVED, class FactorGraphType, class VariableIndexStorage> | 
					
						
							|  |  |  |   static typename DERIVED::shared_ptr Combine(const FactorGraphType& factorGraph, | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  |       const VariableIndex<VariableIndexStorage>& variableIndex, const std::vector<size_t>& factors, | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |       const std::vector<KEY>& variables, const std::vector<std::vector<size_t> >& variablePositions) { | 
					
						
							|  |  |  |     return typename DERIVED::shared_ptr(new DERIVED(variables.begin(), variables.end())); } | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Create a combined joint factor (new style for EliminationTree). */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   template<class DERIVED> | 
					
						
							|  |  |  |   static typename DERIVED::shared_ptr Combine(const FactorGraph<DERIVED>& factors, const FastMap<Key, std::vector<Key> >& variableSlots); | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * eliminate the first variable involved in this factor | 
					
						
							|  |  |  |    * @return a conditional on the eliminated variable | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   template<class CONDITIONAL> | 
					
						
							|  |  |  |   typename CONDITIONAL::shared_ptr eliminateFirst(); | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2010-10-12 08:14:50 +08:00
										 |  |  |    * eliminate the first nrFrontals frontal variables. | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   template<class CONDITIONAL> | 
					
						
							|  |  |  |   typename BayesNet<CONDITIONAL>::shared_ptr eliminate(size_t nrFrontals = 1); | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Permutes the GaussianFactor, but for efficiency requires the permutation | 
					
						
							|  |  |  |    * to already be inverted. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   void permuteWithInverse(const Permutation& inversePermutation); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** iterators */ | 
					
						
							|  |  |  |   const_iterator begin() const { return keys_.begin(); } | 
					
						
							|  |  |  |   const_iterator end() const { return keys_.end(); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** mutable iterators */ | 
					
						
							|  |  |  |   iterator begin() { return keys_.begin(); } | 
					
						
							|  |  |  |   iterator end() { return keys_.end(); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** First key*/ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   Key front() const { return keys_.front(); } | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Last key */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   Key back() const { return keys_.back(); } | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** find */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   const_iterator find(Key key) const { return std::find(begin(), end(), key); } | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** print */ | 
					
						
							|  |  |  |   void print(const std::string& s = "Factor") const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** check equality */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  | //  template<class DERIVED>
 | 
					
						
							|  |  |  |   bool equals(const This& other, double tol = 1e-9) const; | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * return keys in order as created | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2010-10-20 05:31:13 +08:00
										 |  |  |   const std::vector<Key>& keys() const { return keys_; } | 
					
						
							| 
									
										
										
										
											2010-10-09 06:04:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * @return the number of nodes the factor connects | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   size_t size() const { return keys_.size(); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** Serialization function */ | 
					
						
							|  |  |  |   friend class boost::serialization::access; | 
					
						
							|  |  |  |   template<class Archive> | 
					
						
							|  |  |  |   void serialize(Archive & ar, const unsigned int version) { | 
					
						
							|  |  |  |     ar & BOOST_SERIALIZATION_NVP(keys_); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | } |