// @(#)root/roostats:$Id$ // Authors: Kevin Belasco 17/06/2009 // Authors: Kyle Cranmer 17/06/2009 /************************************************************************* * Copyright (C) 1995-2008, Rene Brun and Fons Rademakers. * * All rights reserved. * * * * For the licensing terms see $ROOTSYS/LICENSE. * * For the list of contributors see $ROOTSYS/README/CREDITS. * *************************************************************************/ //_________________________________________________ /* BEGIN_HTML

MCMCCalculator is a concrete implementation of IntervalCalculator. It uses a MetropolisHastings object to construct a Markov Chain of data points in the parameter space. From this Markov Chain, this class can generate a MCMCInterval as per user specification.

The interface allows one to pass the model, data, and parameters via a workspace and then specify them with names.

After configuring the calculator, one only needs to ask GetInterval(), which will return an ConfInterval (MCMCInterval in this case).

END_HTML */ //_________________________________________________ #ifndef ROOT_Rtypes #include "Rtypes.h" #endif #ifndef ROO_GLOBAL_FUNC #include "RooGlobalFunc.h" #endif #ifndef ROO_ABS_REAL #include "RooAbsReal.h" #endif #ifndef ROO_ARG_SET #include "RooArgSet.h" #endif #ifndef ROO_ARG_LIST #include "RooArgList.h" #endif #ifndef ROOSTATS_ModelConfig #include "RooStats/ModelConfig.h" #endif #ifndef RooStats_RooStatsUtils #include "RooStats/RooStatsUtils.h" #endif #ifndef ROOSTATS_MCMCCalculator #include "RooStats/MCMCCalculator.h" #endif #ifndef ROOSTATS_MetropolisHastings #include "RooStats/MetropolisHastings.h" #endif #ifndef ROOSTATS_MarkovChain #include "RooStats/MarkovChain.h" #endif #ifndef RooStats_MCMCInterval #include "RooStats/MCMCInterval.h" #endif #ifndef ROOT_TIterator #include "TIterator.h" #endif #ifndef ROOSTATS_UniformProposal #include "RooStats/UniformProposal.h" #endif #ifndef ROOSTATS_PdfProposal #include "RooStats/PdfProposal.h" #endif #ifndef ROO_PROD_PDF #include "RooProdPdf.h" #endif ClassImp(RooStats::MCMCCalculator); using namespace RooFit; using namespace RooStats; using namespace std; // default constructor MCMCCalculator::MCMCCalculator() : fPropFunc(0), fPdf(0), fPriorPdf(0), fData(0), fAxes(0) { fNumIters = 0; fNumBurnInSteps = 0; fNumBins = 0; fUseKeys = kFALSE; fUseSparseHist = kFALSE; fSize = -1; fIntervalType = MCMCInterval::kShortest; fLeftSideTF = -1; fEpsilon = -1; fDelta = -1; } // constructor from a Model Config with a basic settings package configured // by SetupBasicUsage() MCMCCalculator::MCMCCalculator(RooAbsData& data, const ModelConfig & model) : fPropFunc(0), fData(&data), fAxes(0) { SetModel(model); SetupBasicUsage(); } void MCMCCalculator::SetModel(const ModelConfig & model) { // set the model fPdf = model.GetPdf(); fPriorPdf = model.GetPriorPdf(); fPOI.removeAll(); fNuisParams.removeAll(); fConditionalObs.removeAll(); if (model.GetParametersOfInterest()) fPOI.add(*model.GetParametersOfInterest()); if (model.GetNuisanceParameters()) fNuisParams.add(*model.GetNuisanceParameters()); if (model.GetConditionalObservables()) fConditionalObs.add( *(model.GetConditionalObservables() ) ); } // Constructor for automatic configuration with basic settings. Uses a // UniformProposal, 10,000 iterations, 40 burn in steps, 50 bins for each // RooRealVar, determines interval by histogram. Finds a 95% confidence // interval. void MCMCCalculator::SetupBasicUsage() { fPropFunc = 0; fNumIters = 10000; fNumBurnInSteps = 40; fNumBins = 50; fUseKeys = kFALSE; fUseSparseHist = kFALSE; SetTestSize(0.05); fIntervalType = MCMCInterval::kShortest; fLeftSideTF = -1; fEpsilon = -1; fDelta = -1; } void MCMCCalculator::SetLeftSideTailFraction(Double_t a) { if (a < 0 || a > 1) { coutE(InputArguments) << "MCMCCalculator::SetLeftSideTailFraction: " << "Fraction must be in the range [0, 1]. " << a << "is not allowed." << endl; return; } fLeftSideTF = a; fIntervalType = MCMCInterval::kTailFraction; } MCMCInterval* MCMCCalculator::GetInterval() const { // Main interface to get a RooStats::ConfInterval. if (!fData || !fPdf ) return 0; if (fPOI.getSize() == 0) return 0; if (fSize < 0) { coutE(InputArguments) << "MCMCCalculator::GetInterval: " << "Test size/Confidence level not set. Returning NULL." << endl; return NULL; } // if a proposal funciton has not been specified create a default one bool useDefaultPropFunc = (fPropFunc == 0); bool usePriorPdf = (fPriorPdf != 0); if (useDefaultPropFunc) fPropFunc = new UniformProposal(); // if prior is given create product RooAbsPdf * prodPdf = fPdf; if (usePriorPdf) { TString prodName = TString("product_") + TString(fPdf->GetName()) + TString("_") + TString(fPriorPdf->GetName() ); prodPdf = new RooProdPdf(prodName,prodName,RooArgList(*fPdf,*fPriorPdf) ); } RooArgSet* constrainedParams = prodPdf->getParameters(*fData); RooAbsReal* nll = prodPdf->createNLL(*fData, Constrain(*constrainedParams),ConditionalObservables(fConditionalObs)); delete constrainedParams; RooArgSet* params = nll->getParameters(*fData); RemoveConstantParameters(params); if (fNumBins > 0) { SetBins(*params, fNumBins); SetBins(fPOI, fNumBins); if (dynamic_cast(fPropFunc)) { RooArgSet* proposalVars = ((PdfProposal*)fPropFunc)->GetPdf()-> getParameters((RooAbsData*)NULL); SetBins(*proposalVars, fNumBins); } } MetropolisHastings mh; mh.SetFunction(*nll); mh.SetType(MetropolisHastings::kLog); mh.SetSign(MetropolisHastings::kNegative); mh.SetParameters(*params); if (fChainParams.getSize() > 0) mh.SetChainParameters(fChainParams); mh.SetProposalFunction(*fPropFunc); mh.SetNumIters(fNumIters); MarkovChain* chain = mh.ConstructChain(); TString name = TString("MCMCInterval_") + TString(GetName() ); MCMCInterval* interval = new MCMCInterval(name, fPOI, *chain); if (fAxes != NULL) interval->SetAxes(*fAxes); if (fNumBurnInSteps > 0) interval->SetNumBurnInSteps(fNumBurnInSteps); interval->SetUseKeys(fUseKeys); interval->SetUseSparseHist(fUseSparseHist); interval->SetIntervalType(fIntervalType); if (fIntervalType == MCMCInterval::kTailFraction) interval->SetLeftSideTailFraction(fLeftSideTF); if (fEpsilon >= 0) interval->SetEpsilon(fEpsilon); if (fDelta >= 0) interval->SetDelta(fDelta); interval->SetConfidenceLevel(1.0 - fSize); if (useDefaultPropFunc) delete fPropFunc; if (usePriorPdf) delete prodPdf; delete nll; delete params; return interval; }