FastJet 3.4.0
Loading...
Searching...
No Matches
09-user_info.cc
Go to the documentation of this file.
1//----------------------------------------------------------------------
2/// \file
3/// \page Example09 09 - adding user information to a fastjet::PseudoJet
4///
5/// This example illustrates how it is possible to associate
6/// user-defined information to a fastjet::PseudoJet (beyond the
7/// simple user index), using a class derived from
8/// fastjet::UserInfoBase.
9///
10/// Note that in this example we have chosen to use this
11/// user-defined information to obtain properties of the constituents
12/// of the reconstructed jet (e.g. if the event is made of a hard
13/// interaction and pileup, what part of the reconstructed jet comes
14/// from the hard interaction). To do that, we also show how to
15/// introduce a user-defined fastjet::Selector. For some applications,
16/// it might also be useful to define new recombination schemes using
17/// the extra information.
18///
19/// run it with : ./09-user_info < data/Pythia-dijet-ptmin100-lhc-pileup-1ev.dat
20///
21/// (Note that this event consists of many sub-events, the first one
22/// being the "hard" interaction and the following being minbias
23/// events composing the pileup. It has the specificity that it also
24/// contains the PDG id of the particles)
25///
26/// Source code: 09-user_info.cc
27//----------------------------------------------------------------------
28
29//STARTHEADER
30// $Id$
31//
32// Copyright (c) 2005-2018, Matteo Cacciari, Gavin P. Salam and Gregory Soyez
33//
34//----------------------------------------------------------------------
35// This file is part of FastJet.
36//
37// FastJet is free software; you can redistribute it and/or modify
38// it under the terms of the GNU General Public License as published by
39// the Free Software Foundation; either version 2 of the License, or
40// (at your option) any later version.
41//
42// The algorithms that underlie FastJet have required considerable
43// development and are described in hep-ph/0512210. If you use
44// FastJet as part of work towards a scientific publication, please
45// include a citation to the FastJet paper.
46//
47// FastJet is distributed in the hope that it will be useful,
48// but WITHOUT ANY WARRANTY; without even the implied warranty of
49// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
50// GNU General Public License for more details.
51//
52// You should have received a copy of the GNU General Public License
53// along with FastJet. If not, see <http://www.gnu.org/licenses/>.
54//----------------------------------------------------------------------
55//ENDHEADER
56
57#include "fastjet/ClusterSequence.hh"
58#include "fastjet/Selector.hh"
59#include <iostream> // needed for io
60#include <sstream> // needed for io
61#include <cstdio> // needed for io
62
63using namespace std;
64using namespace fastjet;
65
66//------------------------------------------------------------------------
67// the user information
68//
69// To associate extra information to a PseudoJet, one first has to
70// create a class, derived from UserInfoBase, that contains
71// that information.
72//
73// In our simple example, we shall use 2 informations
74// - the PDG id associated with the particle
75// - the "vertex number" associated with the particle
76class MyUserInfo : public PseudoJet::UserInfoBase{
77public:
78 // default ctor
79 // - pdg_id the PDG id of the particle
80 // - vertex_number theid of the vertex it originates from
81 MyUserInfo(const int & pdg_id_in, const int & vertex_number_in) :
82 _pdg_id(pdg_id_in), _vertex_number(vertex_number_in){}
83
84 /// access to the PDG id
85 int pdg_id() const { return _pdg_id;}
86
87 /// access to the vertex number
88 int vertex_number() const { return _vertex_number;}
89
90protected:
91 int _pdg_id; // the associated pdg id
92 int _vertex_number; // the associated vertex number
93};
94
95
96//------------------------------------------------------------------------
97// Select pi0 and photons
98//
99// This shows how we can build a Selector that uses the user-defined
100// information to select particles that are either pi0's or photons
101// (we choose this purely for simplicity).
102//
103// To create a user-defined Selector, the first step is to
104// create its associated "worker" class, i.e. to derive a class from
105// SelectorWorker. Then (see below), we just write a function
106// (SelectorIsPi0Gamma()) that creates a Selector with the
107// appropriate worker class.
108class SW_IsPi0Gamma : public SelectorWorker{
109public:
110 // default ctor
111 SW_IsPi0Gamma(){}
112
113 // the selector's description
114 string description() const{
115 return "neutral pions or photons";
116 }
117
118 // keeps the ones that have the pdg id of the pi0
119 bool pass(const PseudoJet &p) const{
120 // This is how we access the extra information associated with the
121 // particles we test.
122 // p.user_info<MyUserInfo>()
123 // returns a reference to the user-defined information (of type
124 // MyUserInfo, as mentioned explicitly). It ensures automatically
125 // that there is an associated user info compatible with the
126 // requested type (and throws an error if it is not the case)
127 //
128 // We can then access the "pdg_id" member of MyUserInfo to
129 // extract the targeted information.
130 const int & pdgid = p.user_info<MyUserInfo>().pdg_id();
131 return (pdgid == 111) || (pdgid == 22);
132 }
133};
134
135// the function that allows to write simply
136// Selector sel = SelectorIsPi0Gamma();
137Selector SelectorIsPi0Gamma(){
138 return Selector(new SW_IsPi0Gamma());
139}
140
141//------------------------------------------------------------------------
142// Select particles from a given vertex number
143//
144// This is the same kind of construct as just above except that we
145// select on particles that are originated from a given vertex. The
146// test event has been structured as a superposition of sub-events
147// (the 0th being the hard interaction) and each particle will be
148// associated a vertex number. This Selector allows to select
149// particles corresponding to a given vertex number.
150//
151// As in the previous case, we start with the worker class and then
152// write a function for the Selector itself.
153class SW_VertexNumber : public SelectorWorker{
154public:
155 // ctor from the vertex we want to keep
156 SW_VertexNumber(const int & vertex_number) : _vertex_number(vertex_number){}
157
158 // the selector's description
159 string description() const{
160 ostringstream oss;
161 oss << "vertex number " << _vertex_number;
162 return oss.str();
163 }
164
165 // keeps the ones that have the correct vertex number
166 bool pass(const PseudoJet &p) const{
167 // This is how we access the extra information associated with the
168 // particles we test.
169 // p.user_info<MyUserInfo>()
170 // returns a reference to the user-defined information (of type
171 // MyUserInfo, as mentioned explicitly). It ensures automatically
172 // that there is an associated user info compatible with the
173 // requested type (and throws an error if it is not the case)
174 //
175 // We can then access the "vertex_number" member of MyUserInfo to
176 // extract the targeted information.
177 return p.user_info<MyUserInfo>().vertex_number()==_vertex_number;
178 }
179
180private:
181 int _vertex_number; // the vertex number we're selecting
182};
183
184// The function that allows to write e.g.
185// Selector sel = !SelectorVertexNumber(0);
186// to select particles from all vertices except the 0th.
187Selector SelectorVertexNumber(const int & vertex_number){
188 return Selector(new SW_VertexNumber(vertex_number));
189}
190
191
192//------------------------------------------------------------------------
193// The example code associating user-info to the particles in the event
194int main(){
195 // read in input particles
196 //----------------------------------------------------------
197 vector<PseudoJet> input_particles;
198
199 double px, py , pz, E;
200 string str;
201 int vertex_number=-1;
202 int pdg_id = 21;
203 while (getline(cin, str)){
204 // if the line does not start with #, it's a particle
205 // read its momentum and pdg id
206 if (str[0] != '#'){
207 istringstream iss(str);
208 if (! (iss >> px >> py >> pz >> E >> pdg_id)){
209 cerr << "Wrong file format: particles must be specified with" << endl;
210 cerr << " px py pz E pdg_id" << endl;
211 }
212
213 // first create a PseudoJet with the correct momentum
214 PseudoJet p(px,py,pz,E);
215
216 // associate to that our user-defined extra information
217 // which is done using
218 // PseudoJet::set_user_info()
219 //
220 // IMPORTANT NOTE: set_user_info(...) takes a pointer as an
221 // argument. It will "own" that pointer i.e. will delete it when
222 // all the PseudoJet's using it will be deleted.
223 //
224 // NB: once you've done p.set_user_info(my_user_info_ptr), you must
225 // not call p2.set_user_info(my_user_info_ptr) with the same pointer
226 // because p and p2 will both attempt to delete it when they go out
227 // of scope causing a double-free corruption error. Instead do
228 // p2.user_info_shared_ptr() = p.user_info_shared_ptr();
229 p.set_user_info(new MyUserInfo(pdg_id, vertex_number));
230 PseudoJet p2; // defined only to make the above documentation consistent!
231
232 input_particles.push_back(p);
233 continue;
234 }
235
236 // check if we start a new sub-event
237 if ((str.length()>=9) && (str.compare(0,9,"#SUBSTART")==0)){
238 vertex_number++;
239 }
240 }
241
242
243 // create a jet definition:
244 // a jet algorithm with a given radius parameter
245 //----------------------------------------------------------
246 double R = 0.6;
248
249
250 // run the jet clustering with the above jet definition
251 //----------------------------------------------------------
252 ClusterSequence clust_seq(input_particles, jet_def);
253
254
255 // get the resulting jets ordered in pt
256 //----------------------------------------------------------
257 double ptmin = 25.0;
258 vector<PseudoJet> inclusive_jets = sorted_by_pt(clust_seq.inclusive_jets(ptmin));
259
260
261 // tell the user what was done
262 // - the description of the algorithm used
263 // - extract the inclusive jets with pt > 5 GeV
264 // show the output as
265 // {index, rap, phi, pt}
266 //----------------------------------------------------------
267 cout << "Ran " << jet_def.description() << endl;
268
269 // label the columns
270 printf("%5s %15s %15s %15s %15s %15s\n","jet #",
271 "rapidity", "phi", "pt",
272 "pt_hard", "pt_pi0+gamma");
273
274 // a selection on the 1st vertex
275 Selector sel_vtx0 = SelectorVertexNumber(0);
276
277 // a selection on the pi0
278 Selector sel_pi0gamma = SelectorIsPi0Gamma();
279
280 // print out the details for each jet
281 for (unsigned int i = 0; i < inclusive_jets.size(); i++) {
282 const PseudoJet & full = inclusive_jets[i];
283 const vector<PseudoJet> constituents = full.constituents();
284
285 // get the contribution from the 1st vertex
286 PseudoJet hard = join(sel_vtx0(constituents));
287
288 // get the contribution from the pi0's
289 PseudoJet pi0gamma = join(sel_pi0gamma(constituents));
290
291 // print the result
292 printf("%5u %15.8f %15.8f %15.8f %15.8f %15.8f\n", i,
293 full.rap(), full.phi(), full.perp(),
294 hard.perp(), pi0gamma.perp());
295 }
296
297 return 0;
298}
int main()
an example program showing how to use fastjet
Definition 01-basic.cc:50
deals with clustering
class that is intended to hold a full definition of the jet clusterer
a base class to hold extra user information in a PseudoJet
Definition PseudoJet.hh:423
Class to contain pseudojets, including minimal information of use to jet-clustering routines.
Definition PseudoJet.hh:68
virtual std::vector< PseudoJet > constituents() const
retrieve the constituents.
Definition PseudoJet.cc:692
double rap() const
returns the rapidity or some large value when the rapidity is infinite
Definition PseudoJet.hh:138
const L & user_info() const
returns a reference to the dynamic cast conversion of user_info to type L.
Definition PseudoJet.hh:478
double phi() const
returns phi (in the range 0..2pi)
Definition PseudoJet.hh:123
double perp() const
returns the scalar transverse momentum
Definition PseudoJet.hh:158
default selector worker is an abstract virtual base class
Definition Selector.hh:59
virtual std::string description() const
returns a description of the worker
Definition Selector.hh:94
Class that encodes information about cuts and other selection criteria that can be applied to PseudoJ...
Definition Selector.hh:149
the FastJet namespace
@ antikt_algorithm
like the k_t but with distance measures dij = min(1/kti^2,1/ktj^2) Delta R_{ij}^2 / R^2 diB = 1/kti^2
vector< PseudoJet > sorted_by_pt(const vector< PseudoJet > &jets)
return a vector of jets sorted into decreasing kt2
Definition PseudoJet.cc:882