laura is hosted by Hepforge, IPPP Durham
Laura++  3.6.0
A maximum likelihood fitting package for performing Dalitz-plot analysis.
LauIntegrals.cc
Go to the documentation of this file.
1 
2 /*
3 Copyright 2004 University of Warwick
4 
5 Licensed under the Apache License, Version 2.0 (the "License");
6 you may not use this file except in compliance with the License.
7 You may obtain a copy of the License at
8 
9  http://www.apache.org/licenses/LICENSE-2.0
10 
11 Unless required by applicable law or agreed to in writing, software
12 distributed under the License is distributed on an "AS IS" BASIS,
13 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 See the License for the specific language governing permissions and
15 limitations under the License.
16 */
17 
18 /*
19 Laura++ package authors:
20 John Back
21 Paul Harrison
22 Thomas Latham
23 */
24 
29 #include <iostream>
30 using std::cout;
31 using std::endl;
32 
33 #include "LauConstants.hh"
34 #include "LauIntegrals.hh"
35 
36 #include "TMath.h"
37 
38 LauIntegrals::LauIntegrals( Double_t weightsPrecision ) :
39  weightsPrecision_( weightsPrecision )
40 {
41 }
42 
44 {
45 }
46 
48  weightsPrecision_( rhs.weightsPrecision_ )
49 {
50 }
51 
53 {
54  if ( &rhs != this ) {
56  }
57  return *this;
58 }
59 
60 void LauIntegrals::calcGaussLegendreWeights( const Int_t numPoints,
61  std::vector<Double_t>& abscissas,
62  std::vector<Double_t>& weights )
63 {
64  // Calculate the Gauss-Legendre weights that will be used for the
65  // simple Gaussian integration method given the number of points
66 
67  abscissas.clear();
68  weights.clear();
69  abscissas.resize( numPoints );
70  weights.resize( numPoints );
71 
72  Int_t m = ( numPoints + 1 ) / 2;
73  Double_t dnumPoints( numPoints );
74  Double_t dnumPointsPlusHalf = static_cast<Double_t>( numPoints + 0.5 );
75  Int_t i( 0 ), j( 0 );
76  Double_t p1( 0.0 ), p2( 0.0 ), p3( 0.0 ), pp( 0.0 ), z( 0.0 ), zSq( 0.0 ), z1( 0.0 ), di( 0.0 );
77 
78  for ( i = 1; i <= m; i++ ) {
79 
80  di = static_cast<Double_t>( i );
81  z = TMath::Cos( LauConstants::pi * ( di - 0.25 ) / dnumPointsPlusHalf );
82  zSq = z * z;
83 
84  // Starting with the above approximation for the ith root, we enter the
85  // main loop of refinement by Newton's method
86  do {
87  p1 = 1.0;
88  p2 = 0.0;
89 
90  // Calculate the Legendre polynomial at z - recurrence relation
91  for ( j = 1; j <= numPoints; j++ ) {
92  p3 = p2;
93  p2 = p1;
94  p1 = ( ( 2.0 * j - 1.0 ) * z * p2 - ( j - 1.0 ) * p3 ) / static_cast<Double_t>( j );
95  }
96  // p1 = Legendre polynomial. Compute its derivative, pp.
97 
98  pp = dnumPoints * ( z * p1 - p2 ) / ( zSq - 1.0 );
99  z1 = z;
100  z = z1 - p1 / pp; // Newton's method
101  } while ( TMath::Abs( z - z1 ) > weightsPrecision_ );
102 
103  // Scale the root to the desired interval
104  // Remember that the vector entries start with 0, hence i-1 (where first i value is 1)
105  abscissas[i - 1] = z;
106  abscissas[numPoints - i] = abscissas[i - 1]; // Symmetric abscissa
107  weights[i - 1] = 2.0 / ( ( 1.0 - zSq ) * pp * pp );
108  weights[numPoints - i] = weights[i - 1]; // Symmetric weight
109  }
110 }
111 
112 void LauIntegrals::calcGaussHermiteWeights( const Int_t numPoints,
113  std::vector<Double_t>& abscissas,
114  std::vector<Double_t>& weights )
115 {
116  // Calculate the Gauss-Hermite weights that will be used for the
117  // simple Gaussian integration method for a function weighted by an exp(-x**2) term
118  // These weights are common to all integration intervals - therefore they
119  // should only be calculated once - when the constructor is called, for example.
120 
121  abscissas.clear();
122  weights.clear();
123  abscissas.resize( numPoints );
124  weights.resize( numPoints );
125 
126  Int_t m = ( numPoints + 1 ) / 2;
127  Double_t dnumPoints( numPoints );
128  Int_t i( 0 ), j( 0 ), its( 0 );
129  Double_t p1( 0.0 ), p2( 0.0 ), p3( 0.0 ), pp( 0.0 ), z( 0.0 ), z1( 0.0 );
130  Double_t numPointsTerm( 2 * numPoints + 1 );
131 
132  // The roots are symmetric about the origin. Therefore, we only need to find half of them
133  // Loop over the desired roots.
134  for ( i = 1; i <= m; i++ ) {
135 
136  if ( i == 1 ) {
137  // Initial guess for largest root
138  z = TMath::Sqrt( numPointsTerm ) - 1.85575 * TMath::Power( numPointsTerm, -0.16667 );
139  } else if ( i == 2 ) {
140  // Initial guess for second largest root
141  z -= 1.14 * TMath::Power( dnumPoints, 0.426 ) / z;
142  } else if ( i == 3 ) {
143  // Initial guess for third largest root
144  z = 1.86 * z - 0.86 * abscissas[0];
145  } else if ( i == 4 ) {
146  // Initial guess for fourth largest root
147  z = 1.91 * z - 0.91 * abscissas[1];
148  } else {
149  // Initial guess for other roots
150  z = 2.0 * z - abscissas[i - 3];
151  }
152 
153  for ( its = 1; its <= 10; its++ ) {
154  p1 = LauConstants::pim4;
155  p2 = 0.0;
156 
157  for ( j = 1; j <= numPoints; j++ ) {
158  Double_t dj( j );
159  p3 = p2;
160  p2 = p1;
161  p1 = z * TMath::Sqrt( 2.0 / dj ) * p2 - TMath::Sqrt( ( dj - 1.0 ) / dj ) * p3;
162  }
163  // p1 is now the desired Hermite polynomial. Compute its derivative, pp
164  pp = TMath::Sqrt( 2.0 * dnumPoints ) * p2;
165  z1 = z;
166  if ( pp > 1e-10 ) {
167  z = z1 - p1 / pp;
168  }
169  if ( TMath::Abs( z - z1 ) < weightsPrecision_ )
170  break;
171  }
172 
173  // Scale the root to the desired interval
174  // Remember that the vector entries start with 0, hence i-1 (where first i value is 1)
175  abscissas[i - 1] = z;
176  abscissas[numPoints - i] = -z; // Symmetric abscissa
177  weights[i - 1] = 2.0 / ( pp * pp );
178  weights[numPoints - i] = weights[i - 1]; // Symmetric weight
179  }
180 }
Class for performing numerical integration routines.
Definition: LauIntegrals.hh:43
File containing declaration of LauIntegrals class.
const Double_t pi
Pi.
LauIntegrals(Double_t weightsPrecision=1.0e-6)
Constructor.
Definition: LauIntegrals.cc:38
void calcGaussLegendreWeights(const Int_t numPoints, std::vector< Double_t > &abscissas, std::vector< Double_t > &weights)
Calculate the Gauss-Legendre weights.
Definition: LauIntegrals.cc:60
void calcGaussHermiteWeights(const Int_t numPoints, std::vector< Double_t > &abscissas, std::vector< Double_t > &weights)
Calculate the Gauss-Hermite weights.
const Double_t pim4
One over Pi to the one-fourth.
LauIntegrals & operator=(const LauIntegrals &rhs)
Copy assignment operator.
Definition: LauIntegrals.cc:52
File containing LauConstants namespace.
Double_t weightsPrecision_
The precision to which the weights should be calculated.
Definition: LauIntegrals.hh:83
virtual ~LauIntegrals()
Destructor.
Definition: LauIntegrals.cc:43