Creating a custom Fit using Fitter for data on TH2

Hello everyone,

I’m trying to adapt CombinedFit.C (from ROOT examples) to solve my problem of optimisation. I want to fit two data sets (as TH2) to two different functions that have two parameters, both shared between the two functions, by minimisation of chi squared. I found that the optimisation problem that CombinedFit.C solves is like mine, so I started there. However, the fit does not result in reasonable values, and I could not identify if the data from the TH2 isn’t being loaded properly or something else. I was able to take a profile using ProfileX from my data and do the fit properly by loading the data as TGraph, but the profile limits the data that is fitted.

I wanted to know how can this type of fit using Fit::Fitter and data from a TH2 for this project and future ones. I’m adding my code as text here, since the Forum is not letting me upload neither the code nor the auxiliary files for running my code. I hope I can upload the files soon.

Thanks in advance.

ROOT Version: 6.28/06
Platform: Ubuntu 22.04.5 LTS
Compiler: gcc 11.4.0

// Include standard stuff
#include <cstdlib>
#include <cwchar>
#include <fstream>
#include <iostream>
#include <sstream>
#include <stdio.h>
#include <stdlib.h>
#include <string>
#include <sys/stat.h>

// Include ROOT stuff
#include "Math/IFunction.h"
#include "TCanvas.h"
#include "TClonesArray.h"
#include "TCutG.h"
#include "TDirectory.h"
#include "TF1.h"
#include "TF2.h"
#include "TFile.h"
#include "TGraph.h"
#include "TH1.h"
#include "TH2.h"
#include "TLatex.h"
#include "TObject.h"
#include "TPDF.h"
#include "TProfile.h"
#include "TRatioPlot.h"
#include "TTree.h"
#include <Fit/BinData.h>
#include <Fit/Chi2FCN.h>
#include <Fit/Fitter.h>
#include <HFitInterface.h>
#include <Math/WrappedMultiTF1.h>
#include <TCanvas.h>
#include <TH1.h>
#include <TStyle.h>

using namespace std;
TPDF pdff("result.pdf");

TFile *outfile = new TFile("data.root", "READONLY");

TGraph *simul1;
Double_t InterpolatedGraph(Double_t *x, Double_t *par) {
  return par[0] + par[1] * (simul1->Eval(x[0]));
};

TGraph *simul2;
Double_t InterpolatedGraph2(Double_t *x, Double_t *par) {
  return par[0] + par[1] * (simul2->Eval(x[0]));
};

int ipar[2] = {
    0, // intercept common parameter
    1  // slope common parameter
};

// Create the GlobalCHi2 structure

struct GlobalChi2 {
  GlobalChi2(ROOT::Math::IMultiGenFunction &f1,
             ROOT::Math::IMultiGenFunction &f2)
      : fChi2_1(&f1), fChi2_2(&f2) {}

  double operator()(const double *par) const {
    double p[2];
    for (int i = 0; i < 2; ++i)
      p[i] = par[ipar[i]];

    return (*fChi2_1)(p) + (*fChi2_2)(p);
  }

  const ROOT::Math::IMultiGenFunction *fChi2_1;
  const ROOT::Math::IMultiGenFunction *fChi2_2;
};

char filename[4];

int combinedFitTH2() {

  TH2F *htemp1 = (TH2F *)outfile->Get("htemp1");
  TH2F *htemp2 = (TH2F *)outfile->Get("htemp2");

  TCanvas *canvas = new TCanvas("canvas");
  canvas->Print("result.pdf[");

  canvas->SetLogz();

  simul1 = new TGraph("f1.txt", "%lg %lg");
  TF1 *f1 = new TF1("f1", InterpolatedGraph, 0, 200, 2);
  f1->SetParameters(0, 1);
  f1->SetRange(0, 600, 70, 1400);
  f1->SetParName(0, "intercept");
  f1->SetParName(1, "slope");

  simul2 = new TGraph("f2.txt", "%lg %lg");
  TF1 *f2 = new TF1("f2", InterpolatedGraph2, 0, 20, 2);
  f2->SetParameters(0, 1);
  f2->SetRange(0, 0, 20, 200);
  f2->SetParName(0, "intercept");
  f2->SetParName(1, "slope");

  ROOT::Math::WrappedMultiTF1 wf1(*f1, 0);
  ROOT::Math::WrappedMultiTF1 wf2(*f2, 0);

  ROOT::Fit::DataOptions opt;

  // set the data range
  ROOT::Fit::DataRange range1;
  range1.SetRange(0, 70, 600, 1400);
  ROOT::Fit::BinData data1(opt, range1);
  ROOT::Fit::FillData(data1, htemp1);

  ROOT::Fit::DataRange range2;
  range2.SetRange(0, 15, 0, 200);
  ROOT::Fit::BinData data2(opt, range2);
  ROOT::Fit::FillData(data2, htemp2);

  ROOT::Fit::Chi2Function chi2_f1(data1, wf1);
  ROOT::Fit::Chi2Function chi2_f2(data2, wf2);

  GlobalChi2 globalChi2(chi2_f1, chi2_f2);

  ROOT::Fit::Fitter fitter;

  const int Npar = 2;
  double par0[Npar] = {0, 1};

  // create before the parameter settings in order to fix or set range on them
  fitter.Config().SetParamsSettings(Npar, par0);
  // fitter.Config().ParSettings(0).SetLimits(0,200);
  // fitter.Config().ParSettings(1).SetLimits(0,100);

  fitter.Config().MinimizerOptions().SetPrintLevel(0);
  fitter.Config().SetMinimizer("Minuit2", "Migrad");

  // fit FCN function directly
  // (specify optionally data size and flag to indicate that is a chi2 fit)
  fitter.FitFCN(Npar, globalChi2, par0, data1.Size() + data2.Size(), true);
  ROOT::Fit::FitResult result = fitter.Result();
  result.Print(std::cout);

  std::cout << "Combined fit Chi2 = " << result.Chi2() << std::endl;

  // f1->SetFitResult(result, ipar);
  f1->SetLineColor(kBlue);
  f2->SetLineColor(kRed);
  // hSi->GetListOfFunctions()->Add(f1);
  // hSi->Draw();
  // f1->Draw("same");

  htemp1->Draw("colz");
  htemp2->Draw("samecolz");
  f1->Draw("same");
  f2->Draw("same");

  TLatex *tex3 = new TLatex(0, 713.64,
                            Form("Blue: a*f(x) + b, a = %.2f b = %.2f",
                                 f1->GetParameter(1), f1->GetParameter(0)));
  tex3->SetTextSize(0.03);
  tex3->SetLineWidth(1);
  tex3->Draw();

  TLatex *tex1 = new TLatex(0, 753.64,
                            Form("Red: a*f(x) + b, a = %.2f b = %.2f",
                                 f2->GetParameter(1), f2->GetParameter(0)));
  tex1->SetTextSize(0.03);
  tex1->SetLineWidth(1);
  tex1->Draw();

  canvas->Update();
  canvas->Print("result.pdf", "");

  canvas->Print("result.pdf]");
  return 0;
}

Hi @Lyn,

thank you for your question on the forum. I believe now you should be able to upload files, please let me know if not. In the meantime I will tag @moneta to provide more insights on this.

Cheers,
Marta

Thank you @mczurylo

Here are the files.

data.root (12.5 KB)
f1.txt (1.4 KB)
f2.txt (1.4 KB)
combinedFitTH2.C (4.5 KB)

Hi,

Looking at the code, it seems you are creating the BinData classes as 1D. You should pass the dimension (2) in the constructor, see the doc:
https://root.cern/doc/v628/classROOT_1_1Fit_1_1BinData.html#a4d77046d5cc4456eb46834edfea393c8
do:
ROOT::Fit::BinData data1(opt, range1, 0, 2);

Lorenzo

Thanks for the answer @moneta

However, changing what you suggested on BinData hasn’t made any difference…

Here are the results that I get:

Looking into ROOT::Fit::FillData doc, it does not specify that it can take a TH2 but it doesn’t “complain”. It also doesn’t seem to need a dimension in the constructor, but would you think anything else would?

Thanks for your time.