| 
									
										
										
										
											2012-05-23 14:56:01 +01:00
										 |  |  | #include <cassert>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <numeric>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Histogram.h"
 | 
					
						
							|  |  |  | #include "Util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | HistogramSet::HistogramSet (unsigned size, unsigned range) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   size_ = size; | 
					
						
							|  |  |  |   hist_.resize (range, 0); | 
					
						
							|  |  |  |   hist_[0] = size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | HistogramSet::nextHistogram (void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-24 22:55:20 +01:00
										 |  |  |   for (size_t i = hist_.size() - 1; i-- > 0; ) { | 
					
						
							| 
									
										
										
										
											2012-05-23 14:56:01 +01:00
										 |  |  |     if (hist_[i] > 0) { | 
					
						
							|  |  |  |       hist_[i] --; | 
					
						
							|  |  |  |       hist_[i + 1] = maxCount (i + 1); | 
					
						
							|  |  |  |       clearAfter (i + 1); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-05-24 22:55:20 +01:00
										 |  |  |   assert (std::accumulate (hist_.begin(), hist_.end(), 0) | 
					
						
							|  |  |  |       == (int) size_); | 
					
						
							| 
									
										
										
										
											2012-05-23 14:56:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned | 
					
						
							| 
									
										
										
										
											2012-05-24 22:55:20 +01:00
										 |  |  | HistogramSet::operator[] (size_t idx) const | 
					
						
							| 
									
										
										
										
											2012-05-23 14:56:01 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   assert (idx < hist_.size()); | 
					
						
							|  |  |  |   return hist_[idx]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned | 
					
						
							|  |  |  | HistogramSet::nrHistograms (void) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return HistogramSet::nrHistograms (size_, hist_.size()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | HistogramSet::reset (void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   std::fill (hist_.begin() + 1, hist_.end(), 0); | 
					
						
							|  |  |  |   hist_[0] = size_; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | vector<Histogram>  | 
					
						
							|  |  |  | HistogramSet::getHistograms (unsigned N, unsigned R) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   HistogramSet hs (N, R);   | 
					
						
							|  |  |  |   unsigned H = hs.nrHistograms(); | 
					
						
							|  |  |  |   vector<Histogram> histograms; | 
					
						
							|  |  |  |   histograms.reserve (H); | 
					
						
							|  |  |  |   for (unsigned i = 0; i < H; i++) { | 
					
						
							|  |  |  |     histograms.push_back (hs.hist_); | 
					
						
							|  |  |  |     hs.nextHistogram(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return histograms; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned | 
					
						
							|  |  |  | HistogramSet::nrHistograms (unsigned N, unsigned R) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return Util::nrCombinations (N + R - 1, R - 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-24 22:55:20 +01:00
										 |  |  | size_t | 
					
						
							| 
									
										
										
										
											2012-05-23 14:56:01 +01:00
										 |  |  | HistogramSet::findIndex ( | 
					
						
							|  |  |  |     const Histogram& h, | 
					
						
							|  |  |  |     const vector<Histogram>& hists) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   vector<Histogram>::const_iterator it = std::lower_bound ( | 
					
						
							|  |  |  |        hists.begin(), hists.end(), h, std::greater<Histogram>()); | 
					
						
							|  |  |  |   assert (it != hists.end() && *it == h); | 
					
						
							|  |  |  |   return std::distance (hists.begin(), it); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | vector<double> | 
					
						
							|  |  |  | HistogramSet::getNumAssigns (unsigned N, unsigned R) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   HistogramSet hs (N, R); | 
					
						
							|  |  |  |   double N_fac = Util::logFactorial (N); | 
					
						
							|  |  |  |   unsigned H = hs.nrHistograms(); | 
					
						
							|  |  |  |   vector<double> numAssigns; | 
					
						
							|  |  |  |   numAssigns.reserve (H); | 
					
						
							|  |  |  |   for (unsigned h = 0; h < H; h++) { | 
					
						
							|  |  |  |     double prod = 0.0; | 
					
						
							|  |  |  |     for (unsigned r = 0; r < R; r++) { | 
					
						
							|  |  |  |       prod += Util::logFactorial (hs[r]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     double res = N_fac - prod; | 
					
						
							|  |  |  |     numAssigns.push_back (Globals::logDomain ? res : std::exp (res)); | 
					
						
							|  |  |  |     hs.nextHistogram(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return numAssigns; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ostream& operator<< (ostream &os, const HistogramSet& hs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   os << "#" << hs.hist_; | 
					
						
							|  |  |  |   return os; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned | 
					
						
							| 
									
										
										
										
											2012-05-24 22:55:20 +01:00
										 |  |  | HistogramSet::maxCount (size_t idx) const | 
					
						
							| 
									
										
										
										
											2012-05-23 14:56:01 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   unsigned sum = 0; | 
					
						
							| 
									
										
										
										
											2012-05-24 22:55:20 +01:00
										 |  |  |   for (size_t i = 0; i < idx; i++) { | 
					
						
							| 
									
										
										
										
											2012-05-23 14:56:01 +01:00
										 |  |  |     sum += hist_[i]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return size_ - sum; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2012-05-24 22:55:20 +01:00
										 |  |  | HistogramSet::clearAfter (size_t idx) | 
					
						
							| 
									
										
										
										
											2012-05-23 14:56:01 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   std::fill (hist_.begin() + idx + 1, hist_.end(), 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 |