| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * @file ConstrainedLinearFactorGraph.cpp | 
					
						
							|  |  |  |  * @author Alex Cunningham | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <iostream>
 | 
					
						
							|  |  |  | #include "ConstrainedLinearFactorGraph.h"
 | 
					
						
							| 
									
										
										
										
											2009-09-13 12:13:03 +08:00
										 |  |  | #include "FactorGraph-inl.h" // for getOrdering
 | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | using namespace std; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // trick from some reading group
 | 
					
						
							|  |  |  | #define FOREACH_PAIR( KEY, VAL, COL) BOOST_FOREACH (boost::tie(KEY,VAL),COL)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace gtsam { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | ConstrainedLinearFactorGraph::ConstrainedLinearFactorGraph() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | ConstrainedLinearFactorGraph::ConstrainedLinearFactorGraph( | 
					
						
							|  |  |  | 		const LinearFactorGraph& lfg) { | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 	BOOST_FOREACH(LinearFactor::shared_ptr s, lfg) | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 	{	push_back(s); | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | ConstrainedLinearFactorGraph::~ConstrainedLinearFactorGraph() { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | void ConstrainedLinearFactorGraph::push_back_constraint(LinearConstraint::shared_ptr factor) | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 	constraints_.push_back(factor); | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | bool ConstrainedLinearFactorGraph::is_constrained(const std::string& key) const | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 	BOOST_FOREACH(LinearConstraint::shared_ptr e, constraints_) | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 		if (e->involves(key)) return true; | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | bool ConstrainedLinearFactorGraph::equals(const LinearFactorGraph& fg, double tol) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const ConstrainedLinearFactorGraph* p = (const ConstrainedLinearFactorGraph *) &fg; | 
					
						
							|  |  |  | 	if (p == NULL) return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** check equality factors */ | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 	if (constraints_.size() != p->constraints_.size()) return false; | 
					
						
							|  |  |  | 	BOOST_FOREACH(LinearConstraint::shared_ptr ef1, constraints_) | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		bool contains = false; | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 		BOOST_FOREACH(LinearConstraint::shared_ptr ef2, p->constraints_) | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 		if (ef1->equals(*ef2)) | 
					
						
							|  |  |  | 			contains = true; | 
					
						
							|  |  |  | 		if (!contains) return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** default equality check */ | 
					
						
							|  |  |  | 	return LinearFactorGraph::equals(fg, tol); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | ChordalBayesNet::shared_ptr ConstrainedLinearFactorGraph::eliminate(const Ordering& ordering) { | 
					
						
							|  |  |  | 	ChordalBayesNet::shared_ptr cbn (new ChordalBayesNet()); | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	BOOST_FOREACH(string key, ordering) { | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 		// constraints take higher priority in elimination, so check if
 | 
					
						
							|  |  |  | 		// there are constraints first
 | 
					
						
							|  |  |  | 		if (is_constrained(key)) | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 			ConditionalGaussian::shared_ptr ccg = eliminate_constraint(key); | 
					
						
							|  |  |  | 			cbn->insert(key,ccg); | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			ConditionalGaussian::shared_ptr cg = eliminate_one(key); | 
					
						
							|  |  |  | 			cbn->insert(key,cg); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return cbn; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | ConstrainedConditionalGaussian::shared_ptr ConstrainedLinearFactorGraph::eliminate_constraint(const string& key) | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 	// extract all adjacent constraints
 | 
					
						
							|  |  |  | 	vector<LinearConstraint::shared_ptr> constraint_separator = find_constraints_and_remove(key); | 
					
						
							|  |  |  | 	if (constraint_separator.size() == 0) | 
					
						
							|  |  |  | 		throw invalid_argument("No constraints on node " + key); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// split out a constraint to apply
 | 
					
						
							|  |  |  | 	LinearConstraint::shared_ptr constraint = pick_constraint(constraint_separator); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// perform change of variables on the remaining constraints and reinsert
 | 
					
						
							|  |  |  | 	update_constraints(key, constraint_separator, constraint); | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// perform elimination on the constraint itself to get the constrained conditional gaussian
 | 
					
						
							|  |  |  | 	ConstrainedConditionalGaussian::shared_ptr ccg = constraint->eliminate(key); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// perform a change of variables on the linear factors in the separator
 | 
					
						
							|  |  |  | 	LinearFactorSet separator = find_factors_and_remove(key); | 
					
						
							|  |  |  | 	BOOST_FOREACH(LinearFactor::shared_ptr factor, separator) { | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 		// store the block matrices
 | 
					
						
							|  |  |  | 		map<string, Matrix> blocks; | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 		// get the A1 term and reconstruct new_factor without the eliminated block
 | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 		Matrix A1 = factor->get_A(key); | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 		list<string> factor_keys = factor->keys(); | 
					
						
							|  |  |  | 		BOOST_FOREACH(string cur_key, factor_keys) { | 
					
						
							|  |  |  | 			if (cur_key != key) | 
					
						
							|  |  |  | 				blocks.insert(make_pair(cur_key, factor->get_A(cur_key))); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// get T = A1*inv(C1) term
 | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 		Matrix invC1 = inverse(constraint->get_A(key)); | 
					
						
							|  |  |  | 		Matrix T = A1*invC1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// loop over all nodes in separator of constraint
 | 
					
						
							|  |  |  | 		list<string> constraint_keys = constraint->keys(key); | 
					
						
							|  |  |  | 		BOOST_FOREACH(string cur_key, constraint_keys) { | 
					
						
							|  |  |  | 			Matrix Ci = constraint->get_A(cur_key); | 
					
						
							|  |  |  | 			if (cur_key != key && !factor->involves(cur_key)) { | 
					
						
							|  |  |  | 				Matrix Ai = T*Ci; | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 				blocks.insert(make_pair(cur_key, -1 *Ai)); | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 			} else if (cur_key != key) { | 
					
						
							|  |  |  | 				Matrix Ai = factor->get_A(cur_key) - T*Ci; | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 				blocks.insert(make_pair(cur_key, Ai)); | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 		// construct the updated factor
 | 
					
						
							|  |  |  | 		boost::shared_ptr<MutableLinearFactor> new_factor(new MutableLinearFactor); | 
					
						
							|  |  |  | 		string cur_key; Matrix M; | 
					
						
							|  |  |  | 		FOREACH_PAIR(cur_key, M, blocks) { | 
					
						
							|  |  |  | 			new_factor->insert(cur_key, M); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 		// update RHS of updated factor
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 		Vector new_b(A1.size1()); | 
					
						
							|  |  |  | 		if (factor->get_b().size() == 0) | 
					
						
							|  |  |  | 			new_b = -1 * (T * constraint->get_b()); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			new_b = factor->get_b() - T * constraint->get_b(); | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 		new_factor->set_b(new_b); | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 		// insert the new factor into the graph
 | 
					
						
							|  |  |  | 		push_back(new_factor); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 	return ccg; | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | LinearConstraint::shared_ptr ConstrainedLinearFactorGraph::pick_constraint( | 
					
						
							|  |  |  | 		const std::vector<LinearConstraint::shared_ptr>& constraints) const { | 
					
						
							|  |  |  | 	if (constraints.size() == 0) | 
					
						
							|  |  |  | 		throw invalid_argument("Constraint set is empty!"); | 
					
						
							|  |  |  | 	return constraints[0]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | void ConstrainedLinearFactorGraph::update_constraints(const std::string& key, | 
					
						
							|  |  |  | 		const std::vector<LinearConstraint::shared_ptr>& separator, | 
					
						
							|  |  |  | 		const LinearConstraint::shared_ptr& primary) { | 
					
						
							|  |  |  | 	// Implements update for each constraint, where primary is
 | 
					
						
							|  |  |  | 	//    C1*x1 = d - C2*x2 - ...
 | 
					
						
							|  |  |  | 	// and each constraint is
 | 
					
						
							|  |  |  | 	//    A1*x1 + A2*x2 + ... = b;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// extract components from primary
 | 
					
						
							|  |  |  | 	Matrix invC1 = inverse(primary->get_A(key)); | 
					
						
							|  |  |  | 	Vector d = primary->get_b(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// perform transform on each constraint
 | 
					
						
							|  |  |  | 	// constraint c is the one being updated
 | 
					
						
							|  |  |  | 	BOOST_FOREACH(LinearConstraint::shared_ptr updatee, separator) { | 
					
						
							|  |  |  | 		if (!updatee->equals(*primary)) { | 
					
						
							|  |  |  | 			// build transform matrix
 | 
					
						
							|  |  |  | 			Matrix A1 = updatee->get_A(key); | 
					
						
							|  |  |  | 			Matrix T = A1 * invC1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// copy existing nodes into new factor without the eliminated variable
 | 
					
						
							|  |  |  | 			list<string> updatee_keys = updatee->keys(key); | 
					
						
							|  |  |  | 			map<string, Matrix> blocks; | 
					
						
							|  |  |  | 			BOOST_FOREACH(string cur_key, updatee_keys) { | 
					
						
							|  |  |  | 				blocks[cur_key] = updatee->get_A(cur_key); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// loop over all nodes in separator of constraint
 | 
					
						
							|  |  |  | 			list<string> primary_keys = primary->keys(key); // keys that are not key
 | 
					
						
							|  |  |  | 			BOOST_FOREACH(string cur_key, primary_keys) { | 
					
						
							|  |  |  | 				Matrix Ci = primary->get_A(cur_key); | 
					
						
							|  |  |  | 				if (cur_key != key && !updatee->involves(cur_key)) { | 
					
						
							|  |  |  | 					Matrix Ai = T*Ci; | 
					
						
							|  |  |  | 					blocks[cur_key] = -1 * Ai; | 
					
						
							|  |  |  | 				} else if (cur_key != key) { | 
					
						
							|  |  |  | 					Matrix Ai = updatee->get_A(cur_key) - T*Ci; | 
					
						
							|  |  |  | 					blocks[cur_key] = Ai; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			Vector rhs = updatee->get_b() - T * d; | 
					
						
							|  |  |  | 			LinearConstraint::shared_ptr new_constraint(new LinearConstraint(blocks, rhs)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// reinsert into graph
 | 
					
						
							|  |  |  | 			push_back_constraint(new_constraint); | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-10-15 04:39:59 +08:00
										 |  |  | VectorConfig ConstrainedLinearFactorGraph::optimize(const Ordering& ordering) { | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 	ChordalBayesNet::shared_ptr cbn = eliminate(ordering); | 
					
						
							| 
									
										
										
										
											2009-10-15 04:39:59 +08:00
										 |  |  | 	boost::shared_ptr<VectorConfig> newConfig = cbn->optimize(); | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 	return *newConfig; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | std::vector<LinearConstraint::shared_ptr> | 
					
						
							|  |  |  | ConstrainedLinearFactorGraph::find_constraints_and_remove(const string& key) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	vector<LinearConstraint::shared_ptr> found, uninvolved; | 
					
						
							|  |  |  | 	BOOST_FOREACH(LinearConstraint::shared_ptr constraint, constraints_) { | 
					
						
							|  |  |  | 		if (constraint->involves(key)) { | 
					
						
							|  |  |  | 			found.push_back(constraint); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			uninvolved.push_back(constraint); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	constraints_ = uninvolved; | 
					
						
							|  |  |  | 	return found; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | void ConstrainedLinearFactorGraph::print(const std::string& s) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	cout << "ConstrainedFactorGraph: " << s << endl; | 
					
						
							| 
									
										
										
										
											2009-08-31 10:40:26 +08:00
										 |  |  | 	BOOST_FOREACH(LinearFactor::shared_ptr f, factors_) | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		f->print(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 	BOOST_FOREACH(LinearConstraint::shared_ptr f, constraints_) | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		f->print(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | Ordering ConstrainedLinearFactorGraph::getOrdering() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Ordering ord = LinearFactorGraph::getOrdering(); | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 	cout << "ConstrainedLinearFactorGraph::getOrdering() - Not Implemented!" << endl; | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 	//	BOOST_FOREACH(LinearConstraint::shared_ptr e, constraints_)
 | 
					
						
							|  |  |  | 	//		ord.push_back(e->get_key());
 | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 	return ord; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |