Thrust.cc

Go to the documentation of this file.
00001 // -*- C++ -*-
00002 #include "Rivet/Rivet.hh"
00003 #include "Rivet/Projections/Thrust.hh"
00004 #include "Rivet/Tools/Logging.hh"
00005 
00006 
00007 namespace Rivet {
00008 
00009 
00010   void Thrust::calc(const FinalState& fs) {
00011     calc(fs.particles());
00012   }
00013 
00014   void Thrust::calc(const vector<Particle>& fsparticles) {
00015     vector<Vector3> threeMomenta;
00016     threeMomenta.reserve(fsparticles.size());
00017     foreach (const Particle& p, fsparticles) {
00018       const Vector3 p3 = p.momentum().vector3();
00019       threeMomenta.push_back(p3);
00020     }
00021     _calcThrust(threeMomenta);
00022   }
00023 
00024   void Thrust::calc(const vector<FourMomentum>& fsmomenta) {
00025     vector<Vector3> threeMomenta;
00026     threeMomenta.reserve(fsmomenta.size());
00027     foreach (const FourMomentum& v, fsmomenta) {
00028       threeMomenta.push_back(v.vector3());
00029     }
00030     _calcThrust(threeMomenta);
00031   }
00032 
00033   void Thrust::calc(const vector<Vector3>& fsmomenta) {
00034     _calcThrust(fsmomenta);
00035   }
00036 
00037 
00038   /////////////////////////////////////////////////
00039 
00040 
00041 
00042   inline bool mod2Cmp(const Vector3& a, const Vector3& b) {
00043     return a.mod2() > b.mod2();
00044   }
00045 
00046 
00047   // Do the general case thrust calculation
00048   void _calcT(const vector<Vector3>& momenta, double& t, Vector3& taxis) {
00049     /* This function implements the iterative algorithm as described in the
00050      * Pythia manual. We take eight (four) different starting vectors
00051      * constructed from the four (three) leading particles to make sure that
00052      * we don't find a local maximum.
00053      */
00054     vector<Vector3> p = momenta;
00055     assert(p.size() >= 3);
00056     unsigned int n = 3;
00057     if (p.size() == 3) n = 3;
00058     vector<Vector3> tvec;
00059     vector<double> tval;
00060     std::sort(p.begin(), p.end(), mod2Cmp);
00061     for (unsigned int i=0 ; i<pow(2,n-1) ; i++) {
00062       // Create an initial vector from the leading four jets
00063       Vector3 foo(0,0,0);
00064       int sign=i;
00065       for (unsigned int k=0 ; k<n ; k++) {
00066         (sign%2)==1 ? foo+=p[k] : foo-=p[k];
00067         sign/=2;
00068       }
00069       foo=foo.unit();
00070 
00071       // Iterate
00072       double diff=999.;
00073       while (diff>1e-5) {
00074         Vector3 foobar(0,0,0);
00075         for (unsigned int k=0 ; k<p.size() ; k++)
00076           foo.dot(p[k])>0 ? foobar+=p[k] : foobar-=p[k];
00077         diff=(foo-foobar.unit()).mod();
00078         foo=foobar.unit();
00079       }
00080 
00081       // Calculate the thrust value for the vector we found
00082       t=0.;
00083       for (unsigned int k=0 ; k<p.size() ; k++)
00084         t+=fabs(foo.dot(p[k]));
00085 
00086       // Store everything
00087       tval.push_back(t);
00088       tvec.push_back(foo);
00089     }
00090 
00091     // Pick the solution with the largest thrust
00092     t=0.;
00093     for (unsigned int i=0 ; i<tvec.size() ; i++)
00094       if (tval[i]>t){
00095         t=tval[i];
00096         taxis=tvec[i];
00097       }
00098   }
00099 
00100 
00101 
00102   // Do the full calculation
00103   void Thrust::_calcThrust(const vector<Vector3>& fsmomenta) {
00104     // Make a vector of the three-momenta in the final state
00105     double momentumSum(0.0);
00106     foreach (const Vector3& p3, fsmomenta) {
00107       momentumSum += mod(p3);
00108     }
00109     getLog() << Log::DEBUG << "Number of particles = " << fsmomenta.size() << endl;
00110 
00111 
00112     // Clear the caches
00113     _thrusts.clear();
00114     _thrustAxes.clear();
00115 
00116 
00117     // If there are fewer than 2 visible particles, we can't do much
00118     if (fsmomenta.size() < 2) {
00119       for (int i = 0; i < 3; ++i) {
00120         _thrusts.push_back(-1);
00121         _thrustAxes.push_back(Vector3(0,0,0));
00122       }
00123       return;
00124     }
00125 
00126 
00127     // Handle special case of thrust = 1 if there are only 2 particles
00128     if (fsmomenta.size() == 2) {
00129       Vector3 axis(0,0,0);
00130       _thrusts.push_back(1.0);
00131       _thrusts.push_back(0.0);
00132       _thrusts.push_back(0.0);
00133       axis = fsmomenta[0].unit();
00134       if (axis.z() < 0) axis = -axis;
00135       _thrustAxes.push_back(axis);
00136       /// @todo Improve this --- special directions bad...
00137       /// (a,b,c) _|_ 1/(a^2+b^2) (b,-a,0) etc., but which combination minimises error?
00138       if (axis.z() < 0.75)
00139         _thrustAxes.push_back( (axis.cross(Vector3(0,0,1))).unit() );
00140       else
00141         _thrustAxes.push_back( (axis.cross(Vector3(0,1,0))).unit() );
00142       _thrustAxes.push_back( _thrustAxes[0].cross(_thrustAxes[1]) );
00143       return;
00144     }
00145 
00146 
00147 
00148     // Temporary variables for calcs
00149     Vector3 axis(0,0,0);
00150     double val = 0.;
00151 
00152     // Get thrust
00153     _calcT(fsmomenta, val, axis);
00154     getLog() << Log::DEBUG << "Mom sum = " << momentumSum << endl;
00155     _thrusts.push_back(val / momentumSum);
00156     // Make sure that thrust always points along the +ve z-axis.
00157     if (axis.z() < 0) axis = -axis;
00158     axis = axis.unit();
00159     getLog() << Log::DEBUG << "Axis = " << axis << endl;
00160     _thrustAxes.push_back(axis);
00161 
00162     // Get thrust major
00163     vector<Vector3> threeMomenta;
00164     foreach (const Vector3& v, fsmomenta) {
00165       // Get the part of each 3-momentum which is perpendicular to the thrust axis
00166       const Vector3 vpar = dot(v, axis.unit()) * axis.unit();
00167       threeMomenta.push_back(v - vpar);
00168     }
00169     _calcT(threeMomenta, val, axis);
00170     _thrusts.push_back(val / momentumSum);
00171     if (axis.x() < 0) axis = -axis;
00172     axis = axis.unit();
00173     _thrustAxes.push_back(axis);
00174 
00175     // Get thrust minor
00176     if (_thrustAxes[0].dot(_thrustAxes[1]) < 1e-10) {
00177       axis = _thrustAxes[0].cross(_thrustAxes[1]);
00178       _thrustAxes.push_back(axis);
00179       val = 0.0;
00180       foreach (const Vector3& v, fsmomenta) {
00181         val += fabs(dot(axis, v));
00182       }
00183       _thrusts.push_back(val / momentumSum);
00184     } else {
00185       _thrusts.push_back(-1.0);
00186       _thrustAxes.push_back(Vector3(0,0,0));
00187     }
00188 
00189   }
00190 
00191 
00192 
00193 }