gtsam/cpp/GaussianFactor.h

281 lines
8.2 KiB
C
Raw Normal View History

2009-08-22 06:23:24 +08:00
/**
* @file GaussianFactor.h
2009-08-22 06:23:24 +08:00
* @brief Linear Factor....A Gaussian
* @brief linearFactor
* @author Christian Potthast
*/
// \callgraph
#pragma once
#include <boost/shared_ptr.hpp>
#include <boost/tuple/tuple.hpp>
#include <boost/serialization/map.hpp>
2009-08-22 06:23:24 +08:00
#include "Factor.h"
#include "Matrix.h"
#include "VectorConfig.h"
2009-08-22 06:23:24 +08:00
namespace gtsam {
class GaussianConditional;
class Ordering;
2009-08-22 06:23:24 +08:00
/**
* Base Class for a linear factor.
* GaussianFactor is non-mutable (all methods const!).
2009-08-22 06:23:24 +08:00
* The factor value is exp(-0.5*||Ax-b||^2)
*/
class GaussianFactor: public Factor<VectorConfig> {
2009-08-22 06:23:24 +08:00
public:
typedef boost::shared_ptr<GaussianFactor> shared_ptr;
2009-08-22 06:23:24 +08:00
typedef std::map<std::string, Matrix>::iterator iterator;
typedef std::map<std::string, Matrix>::const_iterator const_iterator;
protected:
std::map<std::string, Matrix> As_; // linear matrices
Vector b_; // right-hand-side
Vector sigmas_; // vector of standard deviations for each row in the factor
2009-08-22 06:23:24 +08:00
public:
// TODO: eradicate, as implies non-const
GaussianFactor() {
2009-08-22 06:23:24 +08:00
}
/** Construct Null factor */
GaussianFactor(const Vector& b_in) :
b_(b_in), sigmas_(ones(b_in.size())){
2009-08-22 06:23:24 +08:00
}
/** Construct unary factor */
GaussianFactor(const std::string& key1, const Matrix& A1,
2009-12-10 23:34:09 +08:00
const Vector& b, double sigma) :
b_(b), sigmas_(repeat(b.size(),sigma)) {
As_.insert(make_pair(key1, A1));
2009-08-22 06:23:24 +08:00
}
/** Construct unary factor with vector of sigmas*/
GaussianFactor(const std::string& key1, const Matrix& A1,
const Vector& b, const Vector& sigmas) :
b_(b), sigmas_(sigmas) {
As_.insert(make_pair(key1, A1));
}
2009-08-22 06:23:24 +08:00
/** Construct binary factor */
GaussianFactor(const std::string& key1, const Matrix& A1,
const std::string& key2, const Matrix& A2,
2009-12-10 23:34:09 +08:00
const Vector& b, double sigma) :
b_(b), sigmas_(repeat(b.size(),sigma)) {
As_.insert(make_pair(key1, A1));
As_.insert(make_pair(key2, A2));
2009-08-22 06:23:24 +08:00
}
/** Construct ternary factor */
GaussianFactor(const std::string& key1, const Matrix& A1,
const std::string& key2, const Matrix& A2,
const std::string& key3, const Matrix& A3,
2009-12-10 23:34:09 +08:00
const Vector& b, double sigma) :
b_(b), sigmas_(repeat(b.size(),sigma)) {
As_.insert(make_pair(key1, A1));
As_.insert(make_pair(key2, A2));
As_.insert(make_pair(key3, A3));
2009-08-22 06:23:24 +08:00
}
/** Construct an n-ary factor */
GaussianFactor(const std::vector<std::pair<std::string, Matrix> > &terms,
2009-12-10 23:34:09 +08:00
const Vector &b, double sigma) :
b_(b), sigmas_(repeat(b.size(),sigma)) {
for(unsigned int i=0; i<terms.size(); i++)
As_.insert(terms[i]);
}
/** Construct an n-ary factor with a multiple sigmas*/
GaussianFactor(const std::vector<std::pair<std::string, Matrix> > &terms,
2009-12-10 23:34:09 +08:00
const Vector &b, const Vector& sigmas) :
b_(b), sigmas_(sigmas) {
for (unsigned int i = 0; i < terms.size(); i++)
As_.insert(terms[i]);
}
2009-08-22 06:23:24 +08:00
/** Construct from Conditional Gaussian */
GaussianFactor(const boost::shared_ptr<GaussianConditional>& cg);
2009-08-22 06:23:24 +08:00
/**
* Constructor that combines a set of factors
* @param factors Set of factors to combine
*/
GaussianFactor(const std::vector<shared_ptr> & factors);
// Implementing Testable virtual functions
void print(const std::string& s = "") const;
bool equals(const Factor<VectorConfig>& lf, double tol = 1e-9) const;
2009-08-22 06:23:24 +08:00
// Implementing Factor virtual functions
2009-12-12 01:42:54 +08:00
Vector error_vector(const VectorConfig& c) const; /** (A*x-b)/sigma */
double error(const VectorConfig& c) const; /** 0.5*(A*x-b)'*D*(A*x-b) */
std::size_t size() const { return As_.size();}
2009-08-22 06:23:24 +08:00
/** STL like, return the iterator pointing to the first node */
const_iterator const begin() const { return As_.begin();}
2009-08-22 06:23:24 +08:00
/** STL like, return the iterator pointing to the last node */
const_iterator const end() const { return As_.end(); }
2009-08-22 06:23:24 +08:00
/** check if empty */
bool empty() const { return b_.size() == 0;}
2009-08-22 06:23:24 +08:00
/** get a copy of b */
const Vector& get_b() const { return b_; }
/** get a copy of sigmas */
const Vector& get_sigmas() const { return sigmas_; }
2009-08-22 06:23:24 +08:00
/**
* get a copy of the A matrix from a specific node
* O(log n)
*/
const Matrix& get_A(const std::string& key) const {
const_iterator it = As_.find(key);
if (it == As_.end())
throw(std::invalid_argument("GaussianFactor::[] invalid key: " + key));
2009-08-22 06:23:24 +08:00
return it->second;
}
/** operator[] syntax for get */
inline const Matrix& operator[](const std::string& name) const {
return get_A(name);
}
/** Check if factor involves variable with key */
bool involves(const std::string& key) const {
const_iterator it = As_.find(key);
return (it != As_.end());
2009-08-22 06:23:24 +08:00
}
/**
* return the number of rows from the b vector
* @return a integer with the number of rows from the b vector
*/
int numberOfRows() const { return b_.size();}
2009-08-22 06:23:24 +08:00
2009-09-12 11:50:44 +08:00
/**
* Find all variables
* @return The set of all variable keys
*/
std::list<std::string> keys() const;
2009-09-12 11:50:44 +08:00
2009-08-22 06:23:24 +08:00
/**
* Find all variables and their dimensions
* @return The set of all variable/dimension pairs
*/
Dimensions dimensions() const;
2009-08-22 06:23:24 +08:00
/**
* Get the dimension of a particular variable
* @param key is the name of the variable
* @return the size of the variable
*/
size_t getDim(const std::string& key) const;
2009-08-22 06:23:24 +08:00
/**
* Add to separator set if this factor involves key, but don't add key itself
* @param key
* @param separator set to add to
*/
void tally_separator(const std::string& key,
std::set<std::string>& separator) const;
/**
* Return (dense) matrix associated with factor
* @param ordering of variables needed for matrix column order
* @param set weight to true to bake in the weights
2009-08-22 06:23:24 +08:00
*/
std::pair<Matrix, Vector> matrix(const Ordering& ordering, bool weight = true) const;
2009-08-22 06:23:24 +08:00
/**
* Return (dense) matrix associated with factor
* The returned system is an augmented matrix: [A b]
* The standard deviations are NOT baked into A and b
* @param ordering of variables needed for matrix column order
*/
Matrix matrix_augmented(const Ordering& ordering) const;
/**
* Return vectors i, j, and s to generate an m-by-n sparse matrix
* such that S(i(k),j(k)) = s(k), which can be given to MATLAB's sparse.
* As above, the standard deviations are baked into A and b
* @param ordering of variables needed for matrix column order
*/
boost::tuple<std::list<int>, std::list<int>, std::list<double> >
sparse(const Ordering& ordering, const Dimensions& variables) const;
/**
* Add gradient contribution to gradient config g
* @param x: confif at which to evaluate gradient
* @param g: I/O parameter, evolving gradient
*/
void addGradientContribution(const VectorConfig& x, VectorConfig& g) const;
/**
* Create a GaussianFactor on one variable 'alpha' (step size), in direction d
* @param x: starting point for search
* @param d: search direction
*/
shared_ptr alphaFactor(const VectorConfig& x, const VectorConfig& d) const;
/* ************************************************************************* */
// MUTABLE functions. FD:on the path to being eradicated
/* ************************************************************************* */
2009-08-22 06:23:24 +08:00
/** insert, copies A */
void insert(const std::string& key, const Matrix& A) {
As_.insert(std::make_pair(key, A));
2009-08-22 06:23:24 +08:00
}
/** set RHS, copies b */
void set_b(const Vector& b) {
this->b_ = b;
2009-08-22 06:23:24 +08:00
}
// set A matrices for the linear factor, same as insert ?
inline void set_A(const std::string& key, const Matrix &A) {
insert(key, A);
}
/**
* Current Implementation: Full QR factorization
2009-08-22 06:23:24 +08:00
* eliminate (in place!) one of the variables connected to this factor
* @param key the key of the node to be eliminated
* @return a new factor and a conditional gaussian on the eliminated variable
*/
std::pair<boost::shared_ptr<GaussianConditional>, shared_ptr>
2009-11-12 12:53:28 +08:00
eliminate(const std::string& key) const;
2009-08-22 06:23:24 +08:00
/**
* Take the factor f, and append to current matrices. Not very general.
* @param f linear factor graph
* @param m final number of rows of f, needs to be known in advance
* @param pos where to insert in the m-sized matrices
*/
void append_factor(GaussianFactor::shared_ptr f, size_t m, size_t pos);
}; // GaussianFactor
/* ************************************************************************* */
2009-08-22 06:23:24 +08:00
/**
* creates a C++ string a la "x3", "m768"
* @param c the base character
* @param index the integer to be added
* @return a C++ string
*/
std::string symbol(char c, int index);
2009-08-22 06:23:24 +08:00
} // namespace gtsam