Fork me on GitHub

source: git/external/fastjet/PseudoJet.cc@ 7b0e00c

Last change on this file since 7b0e00c was 273e668, checked in by Pavel Demin <pavel.demin@…>, 10 years ago

upgrade FastJet to version 3.1.0

  • Property mode set to 100644
File size: 31.2 KB
Line 
1//FJSTARTHEADER
2// $Id: PseudoJet.cc 3652 2014-09-03 13:31:13Z salam $
3//
4// Copyright (c) 2005-2014, Matteo Cacciari, Gavin P. Salam and Gregory Soyez
5//
6//----------------------------------------------------------------------
7// This file is part of FastJet.
8//
9// FastJet is free software; you can redistribute it and/or modify
10// it under the terms of the GNU General Public License as published by
11// the Free Software Foundation; either version 2 of the License, or
12// (at your option) any later version.
13//
14// The algorithms that underlie FastJet have required considerable
15// development. They are described in the original FastJet paper,
16// hep-ph/0512210 and in the manual, arXiv:1111.6097. If you use
17// FastJet as part of work towards a scientific publication, please
18// quote the version you use and include a citation to the manual and
19// optionally also to hep-ph/0512210.
20//
21// FastJet is distributed in the hope that it will be useful,
22// but WITHOUT ANY WARRANTY; without even the implied warranty of
23// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
24// GNU General Public License for more details.
25//
26// You should have received a copy of the GNU General Public License
27// along with FastJet. If not, see <http://www.gnu.org/licenses/>.
28//----------------------------------------------------------------------
29//FJENDHEADER
30
31
32#include "fastjet/Error.hh"
33#include "fastjet/PseudoJet.hh"
34#include "fastjet/ClusterSequence.hh"
35#ifndef __FJCORE__
36#include "fastjet/ClusterSequenceAreaBase.hh"
37#endif // __FJCORE__
38#include "fastjet/CompositeJetStructure.hh"
39#include<valarray>
40#include<iostream>
41#include<sstream>
42#include<cmath>
43#include<algorithm>
44#include <cstdarg>
45
46FASTJET_BEGIN_NAMESPACE // defined in fastjet/internal/base.hh
47
48using namespace std;
49
50
51//----------------------------------------------------------------------
52// another constructor...
53PseudoJet::PseudoJet(const double px_in, const double py_in, const double pz_in, const double E_in) {
54
55 _E = E_in ;
56 _px = px_in;
57 _py = py_in;
58 _pz = pz_in;
59
60 this->_finish_init();
61
62 // some default values for the history and user indices
63 _reset_indices();
64
65}
66
67
68//----------------------------------------------------------------------
69/// do standard end of initialisation
70void PseudoJet::_finish_init () {
71 _kt2 = this->px()*this->px() + this->py()*this->py();
72 _phi = pseudojet_invalid_phi;
73 // strictly speaking, _rap does not need initialising, because
74 // it's never used as long as _phi == pseudojet_invalid_phi
75 // (and gets set when _phi is requested). However ATLAS
76 // 2013-03-28 complained that they sometimes have NaN's in
77 // _rap and this interferes with some of their internal validation.
78 // So we initialise it; penalty is about 0.3ns per PseudoJet out of about
79 // 10ns total initialisation time (on a intel Core i7 2.7GHz)
80 _rap = pseudojet_invalid_rap;
81}
82
83//----------------------------------------------------------------------
84void PseudoJet::_set_rap_phi() const {
85
86 if (_kt2 == 0.0) {
87 _phi = 0.0; }
88 else {
89 _phi = atan2(this->py(),this->px());
90 }
91 if (_phi < 0.0) {_phi += twopi;}
92 if (_phi >= twopi) {_phi -= twopi;} // can happen if phi=-|eps<1e-15|?
93 if (this->E() == abs(this->pz()) && _kt2 == 0) {
94 // Point has infinite rapidity -- convert that into a very large
95 // number, but in such a way that different 0-pt momenta will have
96 // different rapidities (so as to lift the degeneracy between
97 // them) [this can be relevant at parton-level]
98 double MaxRapHere = MaxRap + abs(this->pz());
99 if (this->pz() >= 0.0) {_rap = MaxRapHere;} else {_rap = -MaxRapHere;}
100 } else {
101 // get the rapidity in a way that's modestly insensitive to roundoff
102 // error when things pz,E are large (actually the best we can do without
103 // explicit knowledge of mass)
104 double effective_m2 = max(0.0,m2()); // force non tachyonic mass
105 double E_plus_pz = _E + abs(_pz); // the safer of p+, p-
106 // p+/p- = (p+ p-) / (p-)^2 = (kt^2+m^2)/(p-)^2
107 _rap = 0.5*log((_kt2 + effective_m2)/(E_plus_pz*E_plus_pz));
108 if (_pz > 0) {_rap = - _rap;}
109 }
110
111}
112
113
114//----------------------------------------------------------------------
115// return a valarray four-momentum
116valarray<double> PseudoJet::four_mom() const {
117 valarray<double> mom(4);
118 mom[0] = _px;
119 mom[1] = _py;
120 mom[2] = _pz;
121 mom[3] = _E ;
122 return mom;
123}
124
125//----------------------------------------------------------------------
126// Return the component corresponding to the specified index.
127// taken from CLHEP
128double PseudoJet::operator () (int i) const {
129 switch(i) {
130 case X:
131 return px();
132 case Y:
133 return py();
134 case Z:
135 return pz();
136 case T:
137 return e();
138 default:
139 ostringstream err;
140 err << "PseudoJet subscripting: bad index (" << i << ")";
141 throw Error(err.str());
142 }
143 return 0.;
144}
145
146//----------------------------------------------------------------------
147// return the pseudorapidity
148double PseudoJet::pseudorapidity() const {
149 if (px() == 0.0 && py() ==0.0) return MaxRap;
150 if (pz() == 0.0) return 0.0;
151
152 double theta = atan(perp()/pz());
153 if (theta < 0) theta += pi;
154 return -log(tan(theta/2));
155}
156
157//----------------------------------------------------------------------
158// return "sum" of two pseudojets
159PseudoJet operator+ (const PseudoJet & jet1, const PseudoJet & jet2) {
160 //return PseudoJet(jet1.four_mom()+jet2.four_mom());
161 return PseudoJet(jet1.px()+jet2.px(),
162 jet1.py()+jet2.py(),
163 jet1.pz()+jet2.pz(),
164 jet1.E() +jet2.E() );
165}
166
167//----------------------------------------------------------------------
168// return difference of two pseudojets
169PseudoJet operator- (const PseudoJet & jet1, const PseudoJet & jet2) {
170 //return PseudoJet(jet1.four_mom()-jet2.four_mom());
171 return PseudoJet(jet1.px()-jet2.px(),
172 jet1.py()-jet2.py(),
173 jet1.pz()-jet2.pz(),
174 jet1.E() -jet2.E() );
175}
176
177//----------------------------------------------------------------------
178// return the product, coeff * jet
179PseudoJet operator* (double coeff, const PseudoJet & jet) {
180 // see the comment in operator*= about ensuring valid rap phi
181 // before a multiplication to handle case of multiplication by
182 // zero, while maintaining rapidity and phi
183 jet._ensure_valid_rap_phi();
184 //return PseudoJet(coeff*jet.four_mom());
185 // the following code is hopefully more efficient
186 PseudoJet coeff_times_jet(jet);
187 coeff_times_jet *= coeff;
188 return coeff_times_jet;
189}
190
191//----------------------------------------------------------------------
192// return the product, coeff * jet
193PseudoJet operator* (const PseudoJet & jet, double coeff) {
194 return coeff*jet;
195}
196
197//----------------------------------------------------------------------
198// return the ratio, jet / coeff
199PseudoJet operator/ (const PseudoJet & jet, double coeff) {
200 return (1.0/coeff)*jet;
201}
202
203//----------------------------------------------------------------------
204/// multiply the jet's momentum by the coefficient
205void PseudoJet::operator*=(double coeff) {
206 // operator*= aims to maintain the rapidity and azimuth
207 // for the PseudoJet; if they have already been evaluated
208 // this is fine, but if they haven't and coeff is sufficiently
209 // small as to cause a zero or underflow result, then a subsequent
210 // invocation of rap or phi will lead to a non-sensical result.
211 // So, here, we preemptively ensure that rapidity and phi
212 // are correctly cached
213 _ensure_valid_rap_phi();
214 _px *= coeff;
215 _py *= coeff;
216 _pz *= coeff;
217 _E *= coeff;
218 _kt2*= coeff*coeff;
219 // phi and rap are unchanged
220}
221
222//----------------------------------------------------------------------
223/// divide the jet's momentum by the coefficient
224void PseudoJet::operator/=(double coeff) {
225 (*this) *= 1.0/coeff;
226}
227
228
229//----------------------------------------------------------------------
230/// add the other jet's momentum to this jet
231void PseudoJet::operator+=(const PseudoJet & other_jet) {
232 _px += other_jet._px;
233 _py += other_jet._py;
234 _pz += other_jet._pz;
235 _E += other_jet._E ;
236 _finish_init(); // we need to recalculate phi,rap,kt2
237}
238
239
240//----------------------------------------------------------------------
241/// subtract the other jet's momentum from this jet
242void PseudoJet::operator-=(const PseudoJet & other_jet) {
243 _px -= other_jet._px;
244 _py -= other_jet._py;
245 _pz -= other_jet._pz;
246 _E -= other_jet._E ;
247 _finish_init(); // we need to recalculate phi,rap,kt2
248}
249
250//----------------------------------------------------------------------
251bool operator==(const PseudoJet & a, const PseudoJet & b) {
252 if (a.px() != b.px()) return false;
253 if (a.py() != b.py()) return false;
254 if (a.pz() != b.pz()) return false;
255 if (a.E () != b.E ()) return false;
256
257 if (a.user_index() != b.user_index()) return false;
258 if (a.cluster_hist_index() != b.cluster_hist_index()) return false;
259 if (a.user_info_ptr() != b.user_info_ptr()) return false;
260 if (a.structure_ptr() != b.structure_ptr()) return false;
261
262 return true;
263}
264
265//----------------------------------------------------------------------
266// check if the jet has zero momentum
267bool operator==(const PseudoJet & jet, const double val) {
268 if (val != 0)
269 throw Error("comparing a PseudoJet with a non-zero constant (double) is not allowed.");
270 return (jet.px() == 0 && jet.py() == 0 &&
271 jet.pz() == 0 && jet.E() == 0);
272}
273
274
275
276//----------------------------------------------------------------------
277/// transform this jet (given in the rest frame of prest) into a jet
278/// in the lab frame
279//
280// NB: code adapted from that in herwig f77 (checked how it worked
281// long ago)
282PseudoJet & PseudoJet::boost(const PseudoJet & prest) {
283
284 if (prest.px() == 0.0 && prest.py() == 0.0 && prest.pz() == 0.0)
285 return *this;
286
287 double m_local = prest.m();
288 assert(m_local != 0);
289
290 double pf4 = ( px()*prest.px() + py()*prest.py()
291 + pz()*prest.pz() + E()*prest.E() )/m_local;
292 double fn = (pf4 + E()) / (prest.E() + m_local);
293 _px += fn*prest.px();
294 _py += fn*prest.py();
295 _pz += fn*prest.pz();
296 _E = pf4;
297
298 _finish_init(); // we need to recalculate phi,rap,kt2
299 return *this;
300}
301
302
303//----------------------------------------------------------------------
304/// transform this jet (given in lab) into a jet in the rest
305/// frame of prest
306//
307// NB: code adapted from that in herwig f77 (checked how it worked
308// long ago)
309PseudoJet & PseudoJet::unboost(const PseudoJet & prest) {
310
311 if (prest.px() == 0.0 && prest.py() == 0.0 && prest.pz() == 0.0)
312 return *this;
313
314 double m_local = prest.m();
315 assert(m_local != 0);
316
317 double pf4 = ( -px()*prest.px() - py()*prest.py()
318 - pz()*prest.pz() + E()*prest.E() )/m_local;
319 double fn = (pf4 + E()) / (prest.E() + m_local);
320 _px -= fn*prest.px();
321 _py -= fn*prest.py();
322 _pz -= fn*prest.pz();
323 _E = pf4;
324
325 _finish_init(); // we need to recalculate phi,rap,kt2
326 return *this;
327}
328
329
330//----------------------------------------------------------------------
331/// returns true if the momenta of the two input jets are identical
332bool have_same_momentum(const PseudoJet & jeta, const PseudoJet & jetb) {
333 return jeta.px() == jetb.px()
334 && jeta.py() == jetb.py()
335 && jeta.pz() == jetb.pz()
336 && jeta.E() == jetb.E();
337}
338
339//----------------------------------------------------------------------
340void PseudoJet::set_cached_rap_phi(double rap_in, double phi_in) {
341 _rap = rap_in; _phi = phi_in;
342 if (_phi >= twopi) _phi -= twopi;
343 if (_phi < 0) _phi += twopi;
344}
345
346//----------------------------------------------------------------------
347void PseudoJet::reset_momentum_PtYPhiM(double pt_in, double y_in, double phi_in, double m_in) {
348 assert(phi_in < 2*twopi && phi_in > -twopi);
349 double ptm = (m_in == 0) ? pt_in : sqrt(pt_in*pt_in+m_in*m_in);
350 double exprap = exp(y_in);
351 double pminus = ptm/exprap;
352 double pplus = ptm*exprap;
353 double px_local = pt_in*cos(phi_in);
354 double py_local = pt_in*sin(phi_in);
355 reset_momentum(px_local,py_local,0.5*(pplus-pminus),0.5*(pplus+pminus));
356 set_cached_rap_phi(y_in,phi_in);
357}
358
359//----------------------------------------------------------------------
360/// return a pseudojet with the given pt, y, phi and mass
361PseudoJet PtYPhiM(double pt, double y, double phi, double m) {
362 assert(phi < 2*twopi && phi > -twopi);
363 double ptm = (m == 0) ? pt : sqrt(pt*pt+m*m);
364 double exprap = exp(y);
365 double pminus = ptm/exprap;
366 double pplus = ptm*exprap;
367 double px = pt*cos(phi);
368 double py = pt*sin(phi);
369 PseudoJet mom(px,py,0.5*(pplus-pminus),0.5*(pplus+pminus));
370 mom.set_cached_rap_phi(y,phi);
371 return mom;
372 //return PseudoJet(pt*cos(phi), pt*sin(phi), ptm*sinh(y), ptm*cosh(y));
373}
374
375
376//----------------------------------------------------------------------
377// return kt-distance between this jet and another one
378double PseudoJet::kt_distance(const PseudoJet & other) const {
379 //double distance = min(this->kt2(), other.kt2());
380 double distance = min(_kt2, other._kt2);
381 double dphi = abs(phi() - other.phi());
382 if (dphi > pi) {dphi = twopi - dphi;}
383 double drap = rap() - other.rap();
384 distance = distance * (dphi*dphi + drap*drap);
385 return distance;
386}
387
388
389//----------------------------------------------------------------------
390// return squared cylinder (eta-phi) distance between this jet and another one
391double PseudoJet::plain_distance(const PseudoJet & other) const {
392 double dphi = abs(phi() - other.phi());
393 if (dphi > pi) {dphi = twopi - dphi;}
394 double drap = rap() - other.rap();
395 return (dphi*dphi + drap*drap);
396}
397
398//----------------------------------------------------------------------
399/// returns other.phi() - this.phi(), i.e. the phi distance to
400/// other, constrained to be in range -pi .. pi
401double PseudoJet::delta_phi_to(const PseudoJet & other) const {
402 double dphi = other.phi() - phi();
403 if (dphi > pi) dphi -= twopi;
404 if (dphi < -pi) dphi += twopi;
405 return dphi;
406}
407
408
409string PseudoJet::description() const{
410 // the "default" case of a PJ which does not belong to any cluster sequence
411 if (!_structure())
412 return "standard PseudoJet (with no associated clustering information)";
413
414 // for all the other cases, the description comes from the structure
415 return _structure()->description();
416}
417
418
419
420//----------------------------------------------------------------------
421//
422// The following methods access the associated jet structure (if any)
423//
424//----------------------------------------------------------------------
425
426
427//----------------------------------------------------------------------
428// check whether this PseudoJet has an associated parent
429// ClusterSequence
430bool PseudoJet::has_associated_cluster_sequence() const{
431 return (_structure()) && (_structure->has_associated_cluster_sequence());
432}
433
434//----------------------------------------------------------------------
435// get a (const) pointer to the associated ClusterSequence (NULL if
436// inexistent)
437const ClusterSequence* PseudoJet::associated_cluster_sequence() const{
438 if (! has_associated_cluster_sequence()) return NULL;
439
440 return _structure->associated_cluster_sequence();
441}
442
443
444//----------------------------------------------------------------------
445// check whether this PseudoJet has an associated parent
446// ClusterSequence that is still valid
447bool PseudoJet::has_valid_cluster_sequence() const{
448 return (_structure()) && (_structure->has_valid_cluster_sequence());
449}
450
451//----------------------------------------------------------------------
452// If there is a valid cluster sequence associated with this jet,
453// returns a pointer to it; otherwise throws an Error.
454//
455// Open question: should these errors be upgraded to classes of their
456// own so that they can be caught? [Maybe, but later]
457const ClusterSequence * PseudoJet::validated_cs() const {
458 return validated_structure_ptr()->validated_cs();
459}
460
461
462//----------------------------------------------------------------------
463// set the associated structure
464void PseudoJet::set_structure_shared_ptr(const SharedPtr<PseudoJetStructureBase> &structure_in){
465 _structure = structure_in;
466}
467
468//----------------------------------------------------------------------
469// return true if there is some strusture associated with this PseudoJet
470bool PseudoJet::has_structure() const{
471 return _structure();
472}
473
474//----------------------------------------------------------------------
475// return a pointer to the structure (of type
476// PseudoJetStructureBase*) associated with this PseudoJet.
477//
478// return NULL if there is no associated structure
479const PseudoJetStructureBase* PseudoJet::structure_ptr() const {
480 if (!_structure()) return NULL;
481 return _structure();
482}
483
484//----------------------------------------------------------------------
485// return a non-const pointer to the structure (of type
486// PseudoJetStructureBase*) associated with this PseudoJet.
487//
488// return NULL if there is no associated structure
489//
490// Only use this if you know what you are doing. In any case,
491// prefer the 'structure_ptr()' (the const version) to this method,
492// unless you really need a write access to the PseudoJet's
493// underlying structure.
494PseudoJetStructureBase* PseudoJet::structure_non_const_ptr(){
495 if (!_structure()) return NULL;
496 return _structure();
497}
498
499//----------------------------------------------------------------------
500// return a pointer to the structure (of type
501// PseudoJetStructureBase*) associated with this PseudoJet.
502//
503// throw an error if there is no associated structure
504const PseudoJetStructureBase* PseudoJet::validated_structure_ptr() const {
505 if (!_structure())
506 throw Error("Trying to access the structure of a PseudoJet which has no associated structure");
507 return _structure();
508}
509
510//----------------------------------------------------------------------
511// return a reference to the shared pointer to the
512// PseudoJetStructureBase associated with this PseudoJet
513const SharedPtr<PseudoJetStructureBase> & PseudoJet::structure_shared_ptr() const {
514 return _structure;
515}
516
517
518//----------------------------------------------------------------------
519// check if it has been recombined with another PseudoJet in which
520// case, return its partner through the argument. Otherwise,
521// 'partner' is set to 0.
522//
523// false is also returned if this PseudoJet has no associated
524// ClusterSequence
525bool PseudoJet::has_partner(PseudoJet &partner) const{
526 return validated_structure_ptr()->has_partner(*this, partner);
527}
528
529//----------------------------------------------------------------------
530// check if it has been recombined with another PseudoJet in which
531// case, return its child through the argument. Otherwise, 'child'
532// is set to 0.
533//
534// false is also returned if this PseudoJet has no associated
535// ClusterSequence, with the child set to 0
536bool PseudoJet::has_child(PseudoJet &child) const{
537 return validated_structure_ptr()->has_child(*this, child);
538}
539
540//----------------------------------------------------------------------
541// check if it is the product of a recombination, in which case
542// return the 2 parents through the 'parent1' and 'parent2'
543// arguments. Otherwise, set these to 0.
544//
545// false is also returned if this PseudoJet has no parent
546// ClusterSequence
547bool PseudoJet::has_parents(PseudoJet &parent1, PseudoJet &parent2) const{
548 return validated_structure_ptr()->has_parents(*this, parent1, parent2);
549}
550
551//----------------------------------------------------------------------
552// check if the current PseudoJet contains the one passed as
553// argument
554//
555// false is also returned if this PseudoJet has no associated
556// ClusterSequence.
557bool PseudoJet::contains(const PseudoJet &constituent) const{
558 return validated_structure_ptr()->object_in_jet(constituent, *this);
559}
560
561//----------------------------------------------------------------------
562// check if the current PseudoJet is contained the one passed as
563// argument
564//
565// false is also returned if this PseudoJet has no associated
566// ClusterSequence
567bool PseudoJet::is_inside(const PseudoJet &jet) const{
568 return validated_structure_ptr()->object_in_jet(*this, jet);
569}
570
571
572//----------------------------------------------------------------------
573// returns true if the PseudoJet has constituents
574bool PseudoJet::has_constituents() const{
575 return (_structure()) && (_structure->has_constituents());
576}
577
578//----------------------------------------------------------------------
579// retrieve the constituents.
580vector<PseudoJet> PseudoJet::constituents() const{
581 return validated_structure_ptr()->constituents(*this);
582}
583
584
585//----------------------------------------------------------------------
586// returns true if the PseudoJet has support for exclusive subjets
587bool PseudoJet::has_exclusive_subjets() const{
588 return (_structure()) && (_structure->has_exclusive_subjets());
589}
590
591//----------------------------------------------------------------------
592// return a vector of all subjets of the current jet (in the sense
593// of the exclusive algorithm) that would be obtained when running
594// the algorithm with the given dcut.
595//
596// Time taken is O(m ln m), where m is the number of subjets that
597// are found. If m gets to be of order of the total number of
598// constituents in the jet, this could be substantially slower than
599// just getting that list of constituents.
600//
601// an Error is thrown if this PseudoJet has no currently valid
602// associated ClusterSequence
603std::vector<PseudoJet> PseudoJet::exclusive_subjets (const double dcut) const {
604 return validated_structure_ptr()->exclusive_subjets(*this, dcut);
605}
606
607//----------------------------------------------------------------------
608// return the size of exclusive_subjets(...); still n ln n with same
609// coefficient, but marginally more efficient than manually taking
610// exclusive_subjets.size()
611//
612// an Error is thrown if this PseudoJet has no currently valid
613// associated ClusterSequence
614int PseudoJet::n_exclusive_subjets(const double dcut) const {
615 return validated_structure_ptr()->n_exclusive_subjets(*this, dcut);
616}
617
618//----------------------------------------------------------------------
619// return the list of subjets obtained by unclustering the supplied
620// jet down to n subjets (or all constituents if there are fewer
621// than n).
622//
623// requires n ln n time
624//
625// an Error is thrown if this PseudoJet has no currently valid
626// associated ClusterSequence
627std::vector<PseudoJet> PseudoJet::exclusive_subjets_up_to (int nsub) const {
628 return validated_structure_ptr()->exclusive_subjets_up_to(*this, nsub);
629}
630
631//----------------------------------------------------------------------
632// Same as exclusive_subjets_up_to but throws an error if there are
633// fewer than nsub particles in the jet
634std::vector<PseudoJet> PseudoJet::exclusive_subjets (int nsub) const {
635 vector<PseudoJet> subjets = exclusive_subjets_up_to(nsub);
636 if (int(subjets.size()) < nsub) {
637 ostringstream err;
638 err << "Requested " << nsub << " exclusive subjets, but there were only "
639 << subjets.size() << " particles in the jet";
640 throw Error(err.str());
641 }
642 return subjets;
643}
644
645//----------------------------------------------------------------------
646// return the dij that was present in the merging nsub+1 -> nsub
647// subjets inside this jet.
648//
649// an Error is thrown if this PseudoJet has no currently valid
650// associated ClusterSequence
651double PseudoJet::exclusive_subdmerge(int nsub) const {
652 return validated_structure_ptr()->exclusive_subdmerge(*this, nsub);
653}
654
655//----------------------------------------------------------------------
656// return the maximum dij that occurred in the whole event at the
657// stage that the nsub+1 -> nsub merge of subjets occurred inside
658// this jet.
659//
660// an Error is thrown if this PseudoJet has no currently valid
661// associated ClusterSequence
662double PseudoJet::exclusive_subdmerge_max(int nsub) const {
663 return validated_structure_ptr()->exclusive_subdmerge_max(*this, nsub);
664}
665
666
667// returns true if a jet has pieces
668//
669// By default a single particle or a jet coming from a
670// ClusterSequence have no pieces and this methos will return false.
671bool PseudoJet::has_pieces() const{
672 return ((_structure()) && (_structure->has_pieces(*this)));
673}
674
675// retrieve the pieces that make up the jet.
676//
677// By default a jet does not have pieces.
678// If the underlying interface supports "pieces" retrieve the
679// pieces from there.
680std::vector<PseudoJet> PseudoJet::pieces() const{
681 return validated_structure_ptr()->pieces(*this);
682 // if (!has_pieces())
683 // throw Error("Trying to retrieve the pieces of a PseudoJet that has no support for pieces.");
684 //
685 // return _structure->pieces(*this);
686}
687
688
689//----------------------------------------------------------------------
690// the following ones require a computation of the area in the
691// associated ClusterSequence (See ClusterSequenceAreaBase for details)
692//----------------------------------------------------------------------
693
694#ifndef __FJCORE__
695
696//----------------------------------------------------------------------
697// if possible, return a valid ClusterSequenceAreaBase pointer; otherwise
698// throw an error
699const ClusterSequenceAreaBase * PseudoJet::validated_csab() const {
700 const ClusterSequenceAreaBase *csab = dynamic_cast<const ClusterSequenceAreaBase*>(validated_cs());
701 if (csab == NULL) throw Error("you requested jet-area related information, but the PseudoJet does not have associated area information.");
702 return csab;
703}
704
705
706//----------------------------------------------------------------------
707// check if it has a defined area
708bool PseudoJet::has_area() const{
709 //if (! has_associated_cluster_sequence()) return false;
710 if (! has_structure()) return false;
711 return (validated_structure_ptr()->has_area() != 0);
712}
713
714//----------------------------------------------------------------------
715// return the jet (scalar) area.
716// throw an Error if there is no support for area in the associated CS
717double PseudoJet::area() const{
718 return validated_structure_ptr()->area(*this);
719}
720
721//----------------------------------------------------------------------
722// return the error (uncertainty) associated with the determination
723// of the area of this jet.
724// throws an Error if there is no support for area in the associated CS
725double PseudoJet::area_error() const{
726 return validated_structure_ptr()->area_error(*this);
727}
728
729//----------------------------------------------------------------------
730// return the jet 4-vector area
731// throws an Error if there is no support for area in the associated CS
732PseudoJet PseudoJet::area_4vector() const{
733 return validated_structure_ptr()->area_4vector(*this);
734}
735
736//----------------------------------------------------------------------
737// true if this jet is made exclusively of ghosts
738// throws an Error if there is no support for area in the associated CS
739bool PseudoJet::is_pure_ghost() const{
740 return validated_structure_ptr()->is_pure_ghost(*this);
741}
742
743#endif // __FJCORE__
744
745//----------------------------------------------------------------------
746//
747// end of the methods accessing the information in the associated
748// Cluster Sequence
749//
750//----------------------------------------------------------------------
751
752//----------------------------------------------------------------------
753/// provide a meaningful error message for InexistentUserInfo
754PseudoJet::InexistentUserInfo::InexistentUserInfo() : Error("you attempted to perform a dynamic cast of a PseudoJet's extra info, but the extra info pointer was null")
755{}
756
757
758//----------------------------------------------------------------------
759// sort the indices so that values[indices[0..n-1]] is sorted
760// into increasing order
761void sort_indices(vector<int> & indices,
762 const vector<double> & values) {
763 IndexedSortHelper index_sort_helper(&values);
764 sort(indices.begin(), indices.end(), index_sort_helper);
765}
766
767
768
769//----------------------------------------------------------------------
770/// given a vector of values with a one-to-one correspondence with the
771/// vector of objects, sort objects into an order such that the
772/// associated values would be in increasing order
773template<class T> vector<T> objects_sorted_by_values(
774 const vector<T> & objects,
775 const vector<double> & values) {
776
777 assert(objects.size() == values.size());
778
779 // get a vector of indices
780 vector<int> indices(values.size());
781 for (size_t i = 0; i < indices.size(); i++) {indices[i] = i;}
782
783 // sort the indices
784 sort_indices(indices, values);
785
786 // copy the objects
787 vector<T> objects_sorted(objects.size());
788
789 // place the objects in the correct order
790 for (size_t i = 0; i < indices.size(); i++) {
791 objects_sorted[i] = objects[indices[i]];
792 }
793
794 return objects_sorted;
795}
796
797//----------------------------------------------------------------------
798/// return a vector of jets sorted into decreasing kt2
799vector<PseudoJet> sorted_by_pt(const vector<PseudoJet> & jets) {
800 vector<double> minus_kt2(jets.size());
801 for (size_t i = 0; i < jets.size(); i++) {minus_kt2[i] = -jets[i].kt2();}
802 return objects_sorted_by_values(jets, minus_kt2);
803}
804
805//----------------------------------------------------------------------
806/// return a vector of jets sorted into increasing rapidity
807vector<PseudoJet> sorted_by_rapidity(const vector<PseudoJet> & jets) {
808 vector<double> rapidities(jets.size());
809 for (size_t i = 0; i < jets.size(); i++) {rapidities[i] = jets[i].rap();}
810 return objects_sorted_by_values(jets, rapidities);
811}
812
813//----------------------------------------------------------------------
814/// return a vector of jets sorted into decreasing energy
815vector<PseudoJet> sorted_by_E(const vector<PseudoJet> & jets) {
816 vector<double> energies(jets.size());
817 for (size_t i = 0; i < jets.size(); i++) {energies[i] = -jets[i].E();}
818 return objects_sorted_by_values(jets, energies);
819}
820
821//----------------------------------------------------------------------
822/// return a vector of jets sorted into increasing pz
823vector<PseudoJet> sorted_by_pz(const vector<PseudoJet> & jets) {
824 vector<double> pz(jets.size());
825 for (size_t i = 0; i < jets.size(); i++) {pz[i] = jets[i].pz();}
826 return objects_sorted_by_values(jets, pz);
827}
828
829
830
831//-------------------------------------------------------------------------------
832// helper functions to build a jet made of pieces
833//-------------------------------------------------------------------------------
834
835// build a "CompositeJet" from the vector of its pieces
836//
837// In this case, E-scheme recombination is assumed to compute the
838// total momentum
839PseudoJet join(const vector<PseudoJet> & pieces){
840 // compute the total momentum
841 //--------------------------------------------------
842 PseudoJet result; // automatically initialised to 0
843 for (unsigned int i=0; i<pieces.size(); i++)
844 result += pieces[i];
845
846 // attach a CompositeJetStructure to the result
847 //--------------------------------------------------
848 CompositeJetStructure *cj_struct = new CompositeJetStructure(pieces);
849
850 result.set_structure_shared_ptr(SharedPtr<PseudoJetStructureBase>(cj_struct));
851
852 return result;
853}
854
855// build a "CompositeJet" from a single PseudoJet
856PseudoJet join(const PseudoJet & j1){
857 return join(vector<PseudoJet>(1,j1));
858}
859
860// build a "CompositeJet" from two PseudoJet
861PseudoJet join(const PseudoJet & j1, const PseudoJet & j2){
862 vector<PseudoJet> pieces;
863 pieces.reserve(2);
864 pieces.push_back(j1);
865 pieces.push_back(j2);
866 return join(pieces);
867}
868
869// build a "CompositeJet" from 3 PseudoJet
870PseudoJet join(const PseudoJet & j1, const PseudoJet & j2, const PseudoJet & j3){
871 vector<PseudoJet> pieces;
872 pieces.reserve(3);
873 pieces.push_back(j1);
874 pieces.push_back(j2);
875 pieces.push_back(j3);
876 return join(pieces);
877}
878
879// build a "CompositeJet" from 4 PseudoJet
880PseudoJet join(const PseudoJet & j1, const PseudoJet & j2, const PseudoJet & j3, const PseudoJet & j4){
881 vector<PseudoJet> pieces;
882 pieces.reserve(4);
883 pieces.push_back(j1);
884 pieces.push_back(j2);
885 pieces.push_back(j3);
886 pieces.push_back(j4);
887 return join(pieces);
888}
889
890
891
892
893FASTJET_END_NAMESPACE
894
Note: See TracBrowser for help on using the repository browser.