// Author: Stefan Schmitt
// DESY, 10/08/11
// Version 17.5, in parallel to changes in TUnfold
//
// History:
// Version 17.4, bug fix with error handling
// Version 17.3, bug fix with underflow/overflow bins
// Version 17.2, new option isPeriodic
// Version 17.1, in parallel to TUnfold
// Version 17.0, initial version, numbered in parallel to TUnfold
#ifndef ROOT_TUnfoldBinning
#define ROOT_TUnfoldBinning
//////////////////////////////////////////////////////////////////////////
// //
// //
// TUnfoldBinning, an auxillary class to provide //
// complex binning schemes as input to TUnfoldDensity //
// //
// Citation: S.Schmitt, JINST 7 (2012) T10003 [arXiv:1205.6201] //
// //
//////////////////////////////////////////////////////////////////////////
/*
This file is part of TUnfold.
TUnfold is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
TUnfold is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with TUnfold. If not, see .
*/
#include "TUnfold.h"
#include
#include
#include
#include
#include
#include
class TUnfoldBinning : public TNamed {
protected:
/// mother node
TUnfoldBinning *parentNode;
/// first daughter node
TUnfoldBinning *childNode;
/// next sister
TUnfoldBinning *nextNode;
/// previous sister
TUnfoldBinning *prevNode;
/// for each axis the bin borders (TVectorD)
TObjArray *fAxisList;
/// for each axis its name (TObjString), or names of unconnected bins
TObjArray *fAxisLabelList;
/// bit fields indicating whether there are underflow bins on the axes
Int_t fHasUnderflow;
/// bit fields indicating whether there are overflow bins on the axes
Int_t fHasOverflow;
/// number of bins in this node's distribution
Int_t fDistributionSize;
/// global bin number of the first bin
Int_t fFirstBin;
/// global bin number of the last(+1) bin, including daughters
Int_t fLastBin;
/// function to calculate a scale factor from bin centres (may be a TF1 or a TVectorD
TObject *fBinFactorFunction;
/// common scale factor for all bins of this node
Double_t fBinFactorConstant;
public:
/********************* setup **************************/
enum {
/// maximum numner of axes per distribution
MAXDIM=32
};
TUnfoldBinning(const char *name=0,Int_t nBins=0,const char *binNames=0); // create a new root node with a given number of unconnected bins
TUnfoldBinning(const TAxis &axis,Int_t includeUnderflow,Int_t includeOverflow); // create a binning scheme with one axis
TUnfoldBinning *AddBinning
(TUnfoldBinning *binning); // add a new node to the TUnfoldBinning tree
TUnfoldBinning *AddBinning(const char *name,Int_t nBins=0,const char *binNames=0); // add a new node to the TUnfoldBinning tree
Bool_t AddAxis(const char *name,Int_t nBins,const Double_t *binBorders,
Bool_t hasUnderflow,Bool_t hasOverflow); // add an axis (variable bins) to the distribution associated with this node
Bool_t AddAxis(const char *name,Int_t nBins,Double_t xMin,Double_t xMax,
Bool_t hasUnderflow,Bool_t hasOverflow); // add an axis (equidistant bins) to the distribution associated with this node
Bool_t AddAxis(const TAxis &axis,Bool_t includeUnderflow,Bool_t includeOverflow); // add an axis (from TAxis instance) to the distribution associated with this node
virtual ~TUnfoldBinning(void);
void PrintStream(std::ostream &out,Int_t indent=0,int debug=0) const;
void SetBinFactorFunction(Double_t normalisation,TF1 *userFunc=0); // define function to calculate bin factor. Note: the function is not owned by this class
/********************* Navigation **********************/
/// first daughter node
inline TUnfoldBinning const *GetChildNode(void) const { return childNode; }
/// previous sister node
inline TUnfoldBinning const *GetPrevNode(void) const { return prevNode; }
/// next sister node
inline TUnfoldBinning const *GetNextNode(void) const { return nextNode; }
/// mother node
inline TUnfoldBinning const *GetParentNode(void) const { return parentNode; }
TUnfoldBinning const *FindNode(char const *name) const; // find node by name
/// return root node of the binnig scheme
TUnfoldBinning const *GetRootNode(void) const;
/********************* Create THxx histograms **********/
Int_t GetTH1xNumberOfBins(Bool_t originalAxisBinning=kTRUE,const char *axisSteering=0) const; // get number of bins of a one-dimensional histogram TH1
TH1 *CreateHistogram(const char *histogramName,Bool_t originalAxisBinning=kFALSE,Int_t **binMap=0,const char *histogramTitle=0,const char *axisSteering=0) const; // create histogram and bin map for this node
TH2D *CreateErrorMatrixHistogram(const char *histogramName,Bool_t originalAxisBinning,Int_t **binMap=0,const char *histogramTitle=0,const char *axisSteering=0) const; // create histogram and bin map for this node
static TH2D *CreateHistogramOfMigrations(TUnfoldBinning const *xAxis,
TUnfoldBinning const *yAxis,
char const *histogramName,
Bool_t originalXAxisBinning=kFALSE,
Bool_t originalYAxisBinning=kFALSE,
char const *histogramTitle=0); // create 2D histogram with one binning on the x axis and the other binning on the y axis
TH1 *ExtractHistogram(const char *histogramName,const TH1 *globalBins,const TH2 *globalBinsEmatrix=0,Bool_t originalAxisBinning=kTRUE,const char *axisSteering=0) const; // extract a distribution from the given set of global bins
/********************* Create and manipulate bin map ****/
Int_t *CreateEmptyBinMap(void) const; // create empty bin map
void SetBinMapEntry(Int_t *binMap,Int_t globalBin,Int_t destBin) const; // change mapping for a given global bin
Int_t FillBinMap1D(Int_t *binMap,const char *axisSteering,
Int_t firstBinX) const; // map bins from this distribution to destHist
/********************* Calculate global bin number ******/
Int_t GetGlobalBinNumber(Double_t x) const; // get bin number 1-dim distribution
Int_t GetGlobalBinNumber(Double_t x,Double_t y) const; // get bin number 2-dim distribution
Int_t GetGlobalBinNumber(Double_t x,Double_t y,Double_t z) const; // get bin number 3-dim distribution
Int_t GetGlobalBinNumber(Double_t x0,Double_t x1,Double_t x2,Double_t x3) const; // get bin number four-dimensional binning
Int_t GetGlobalBinNumber(Double_t x0,Double_t x1,Double_t x2,Double_t x3,Double_t x4) const; // get bin number five-dimensional binning
Int_t GetGlobalBinNumber(Double_t x0,Double_t x1,Double_t x2,Double_t x3,Double_t x4,Double_t x5) const; // get bin number six-dimensional binning
Int_t GetGlobalBinNumber(const Double_t *x,Int_t *isBelow=0,Int_t *isAbove=0) const; // get bin number, up to 32 dimensional binning
/// first bin of this node
inline Int_t GetStartBin(void) const { return fFirstBin; }
/// last+1 bin of this node (includes children)
inline Int_t GetEndBin(void) const { return fLastBin; }
virtual Bool_t IsBinFactorGlobal(void) const; // check whether global or local bin factor must be used
Double_t GetGlobalFactor(void) const; // return global factor
/********************* access by global bin number ******/
TString GetBinName(Int_t iBin) const; // return bin name
Double_t GetBinSize(Int_t iBin) const; // return bin size (in N dimensions)
virtual Double_t GetBinFactor(Int_t iBin) const; // return user factor
void GetBinUnderflowOverflowStatus(Int_t iBin,Int_t *uStatus,Int_t *oStatus) const; // return bit map indicating underflow and overflow status
Int_t GetBinNeighbours(Int_t globalBin,Int_t axis,
Int_t *prev,Double_t *distPrev,
Int_t *next,Double_t *distNext,
Bool_t isPeriodic=kFALSE) const; // get neighbour bins along an axis
/********************** access distribution properties *************/
/// number of bins in the distribution possibly including under/overflow
inline Int_t GetDistributionNumberOfBins(void) const { return fDistributionSize; }
/// query dimension of this node's distribution
inline Int_t GetDistributionDimension(void) const { return fAxisList->GetEntriesFast(); }
virtual Double_t GetDistributionAverageBinSize(Int_t axis,Bool_t includeUnderflow, Bool_t includeOverflow) const; // get average bin size
/// get vector of bin borders for one axis
inline TVectorD const *GetDistributionBinning(Int_t axis) const {
return (TVectorD const *)fAxisList->At(axis); }
/// get name of an axis
inline TString GetDistributionAxisLabel(Int_t axis) const {
return ((TObjString const *)fAxisLabelList->At(axis))->GetString(); }
virtual Double_t GetDistributionUnderflowBinWidth(Int_t axis) const; // width of underflow bin on the given axis
virtual Double_t GetDistributionOverflowBinWidth(Int_t axis) const; // width of overflow bin on the given axis
virtual Double_t GetDistributionBinCenter(Int_t axis,Int_t bin) const; // position of bin center on the given axis
Bool_t HasUnconnectedBins(void) const; // check whether this node has bins without axis
const TObjString *GetUnconnectedBinName(Int_t bin) const; // return bin name (if any)
/// check whether an axis has an underflow bin
Bool_t HasUnderflow(int axis) const { return fHasUnderflow & (1<