Geant4 9.6.0
Toolkit for the simulation of the passage of particles through matter
Loading...
Searching...
No Matches
G4Element.cc
Go to the documentation of this file.
1//
2// ********************************************************************
3// * License and Disclaimer *
4// * *
5// * The Geant4 software is copyright of the Copyright Holders of *
6// * the Geant4 Collaboration. It is provided under the terms and *
7// * conditions of the Geant4 Software License, included in the file *
8// * LICENSE and available at http://cern.ch/geant4/license . These *
9// * include a list of copyright holders. *
10// * *
11// * Neither the authors of this software system, nor their employing *
12// * institutes,nor the agencies providing financial support for this *
13// * work make any representation or warranty, express or implied, *
14// * regarding this software system or assume any liability for its *
15// * use. Please see the license in the file LICENSE and URL above *
16// * for the full disclaimer and the limitation of liability. *
17// * *
18// * This code implementation is the result of the scientific and *
19// * technical work of the GEANT4 collaboration. *
20// * By using, copying, modifying or distributing the software (or *
21// * any work based on the software) you agree to acknowledge its *
22// * use in resulting scientific publications, and indicate your *
23// * acceptance of all terms of the Geant4 Software license. *
24// ********************************************************************
25//
26//
27// $Id$
28//
29//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
30
31// 26-06-96: Code uses operators (+=, *=, ++, -> etc.) correctly, P. Urban
32// 09-07-96: new data members added by L.Urban
33// 17-01-97: aesthetic rearrangement, M.Maire
34// 20-01-97: Compute Tsai's formula for the rad length, M.Maire
35// 21-01-97: remove mixture flag, M.Maire
36// 24-01-97: ComputeIonisationParameters().
37// new data member: fTaul, M.Maire
38// 29-01-97: Forbidden to create Element with Z<1 or N<Z, M.Maire
39// 20-03-97: corrected initialization of pointers, M.Maire
40// 28-04-98: atomic subshell binding energies stuff, V. Grichine
41// 09-07-98: Ionisation parameters removed from the class, M.Maire
42// 16-11-98: name Subshell -> Shell; GetBindingEnergy() (mma)
43// 09-03-01: assignement operator revised (mma)
44// 02-05-01: check identical Z in AddIsotope (marc)
45// 03-05-01: flux.precision(prec) at begin/end of operator<<
46// 13-09-01: suppression of the data member fIndexInTable
47// 14-09-01: fCountUse: nb of materials which use this element
48// 26-02-02: fIndexInTable renewed
49// 30-03-05: warning in GetElement(elementName)
50// 15-11-05: GetElement(elementName, G4bool warning=true)
51// 17-10-06: Add fNaturalAbandances (V.Ivanchenko)
52// 27-07-07: improve destructor (V.Ivanchenko)
53// 18-10-07: move definition of material index to ComputeDerivedQuantities (VI)
54// 25.10.11: new scheme for G4Exception (mma)
55// 05-03-12: always create isotope vector (V.Ivanchenko)
56
57//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
58
59#include <iomanip>
60#include <sstream>
61
62#include "G4Element.hh"
63#include "G4AtomicShells.hh"
64#include "G4NistManager.hh"
66#include "G4SystemOfUnits.hh"
67
68G4ElementTable G4Element::theElementTable;
69
70//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
71
72// Constructor to Generate an element from scratch
73
74G4Element::G4Element(const G4String& name, const G4String& symbol,
75 G4double zeff, G4double aeff)
76 : fName(name), fSymbol(symbol)
77{
78 G4int iz = (G4int)zeff;
79 if (zeff<1.) {
81 ed << "Fail to create G4Element " << name
82 << " Z= " << zeff << " < 1 !" << G4endl;
83 G4Exception ("G4Element::G4Element()", "mat011", FatalException, ed);
84 }
85 if (std::abs(zeff - iz) > perMillion) {
87 ed << "G4Element Warning: " << name << " Z= " << zeff
88 << " A= " << aeff/(g/mole) << G4endl;
89 G4Exception("G4Element::G4Element()", "mat017", JustWarning, ed);
90 }
91
92 InitializePointers();
93
94 fZeff = zeff;
95 fNeff = aeff/(g/mole);
96 fAeff = aeff;
97
98 if(fNeff < 1.0) fNeff = 1.0;
99
100 if (fNeff < zeff) {
102 ed << "Fail to create G4Element " << name
103 << " with Z= " << zeff << " N= " << fNeff
104 << " N < Z is not allowed" << G4endl;
105 G4Exception ("G4Element::G4Element()", "mat012", FatalException, ed);
106 }
107
108 fNbOfAtomicShells = G4AtomicShells::GetNumberOfShells(iz);
109 fAtomicShells = new G4double[fNbOfAtomicShells];
110 fNbOfShellElectrons = new G4int[fNbOfAtomicShells];
111
112 AddNaturalIsotopes();
113
114 for (G4int i=0;i<fNbOfAtomicShells;i++)
115 {
116 fAtomicShells[i] = G4AtomicShells::GetBindingEnergy(iz, i);
117 fNbOfShellElectrons[i] = G4AtomicShells::GetNumberOfElectrons(iz, i);
118 }
119 ComputeDerivedQuantities();
120}
121
122//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
123
124// Constructor to Generate element from a List of 'nIsotopes' isotopes, added
125// via AddIsotope
126
128 const G4String& symbol, G4int nIsotopes)
129 : fName(name),fSymbol(symbol)
130{
131 InitializePointers();
132
133 size_t n = size_t(nIsotopes);
134
135 if(0 == nIsotopes) {
136 AddNaturalIsotopes();
137 } else {
138 theIsotopeVector = new G4IsotopeVector(n,0);
139 fRelativeAbundanceVector = new G4double[nIsotopes];
140 }
141}
142
143//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
144
145// Add an isotope to the element
146
147void G4Element::AddIsotope(G4Isotope* isotope, G4double abundance)
148{
149 if (theIsotopeVector == 0) {
151 ed << "Fail to add Isotope to G4Element " << fName
152 << " with Z= " << fZeff << " N= " << fNeff << G4endl;
153 G4Exception ("G4Element::AddIsotope()", "mat013", FatalException, ed);
154 return;
155 }
156 G4int iz = isotope->GetZ();
157
158 // filling ...
159 if ( fNumberOfIsotopes < theIsotopeVector->size() ) {
160 // check same Z
161 if (fNumberOfIsotopes==0) { fZeff = G4double(iz); }
162 else if (G4double(iz) != fZeff) {
164 ed << "Fail to add Isotope Z= " << iz << " to G4Element " << fName
165 << " with different Z= " << fZeff << fNeff
166 << G4endl;
167 G4Exception ("G4Element::AddIsotope()", "mat014", FatalException, ed);
168 return;
169 }
170 //Z ok
171 fRelativeAbundanceVector[fNumberOfIsotopes] = abundance;
172 (*theIsotopeVector)[fNumberOfIsotopes] = isotope;
173 ++fNumberOfIsotopes;
174 isotope->increaseCountUse();
175 } else {
177 ed << "Fail to add Isotope Z= " << iz << " to G4Element " << fName
178 << " - more isotopes than declaired " << G4endl;
179 G4Exception ("G4Element::AddIsotope()", "mat015", FatalException, ed);
180 return;
181 }
182
183 // filled.
184 if ( fNumberOfIsotopes == theIsotopeVector->size() ) {
185 // Compute Neff, Aeff
186 G4double wtSum=0.0;
187
188 fNeff = fAeff = 0.0;
189 for (size_t i=0;i<fNumberOfIsotopes;i++) {
190 fAeff += fRelativeAbundanceVector[i]*(*theIsotopeVector)[i]->GetA();
191 fNeff += fRelativeAbundanceVector[i]*(*theIsotopeVector)[i]->GetN();
192 wtSum += fRelativeAbundanceVector[i];
193 }
194 fAeff /= wtSum;
195 fNeff /= wtSum;
196
197 if(wtSum != 1.0) {
198 for(size_t i=0; i<fNumberOfIsotopes; ++i) {
199 fRelativeAbundanceVector[i] /= wtSum;
200 }
201 }
202
203 fNbOfAtomicShells = G4AtomicShells::GetNumberOfShells(iz);
204 fAtomicShells = new G4double[fNbOfAtomicShells];
205 fNbOfShellElectrons = new G4int[fNbOfAtomicShells];
206
207 for ( G4int j = 0; j < fNbOfAtomicShells; j++ )
208 {
209 fAtomicShells[j] = G4AtomicShells::GetBindingEnergy(iz, j);
210 fNbOfShellElectrons[j] = G4AtomicShells::GetNumberOfElectrons(iz, j);
211 }
212 ComputeDerivedQuantities();
213
214 }
215}
216
217//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
218
219void G4Element::InitializePointers()
220{
221 theIsotopeVector = 0;
222 fRelativeAbundanceVector = 0;
223 fAtomicShells = 0;
224 fNbOfShellElectrons = 0;
225 fIonisation = 0;
226 fNumberOfIsotopes = 0;
227 fNaturalAbandances = false;
228
229 // add initialisation of all remaining members
230 fZeff = 0;
231 fNeff = 0;
232 fAeff = 0;
233 fNbOfAtomicShells = 0;
234 fCountUse = 0;
235 fIndexZ = 0;
236 fIndexInTable = 0;
237 fNaturalAbandances = false;
238 fCoulomb = 0.0;
239 fRadTsai = 0.0;
240}
241
242//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
243
244// Fake default constructor - sets only member data and allocates memory
245// for usage restricted to object persistency
246
248 : fZeff(0), fNeff(0), fAeff(0)
249{
250 InitializePointers();
251}
252
253//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
254
256{
257 // G4cout << "### Destruction of element " << fName << " started" <<G4endl;
258
259 if (theIsotopeVector) { delete theIsotopeVector; }
260 if (fRelativeAbundanceVector) { delete [] fRelativeAbundanceVector; }
261 if (fAtomicShells) { delete [] fAtomicShells; }
262 if (fNbOfShellElectrons) { delete [] fNbOfShellElectrons; }
263 if (fIonisation) { delete fIonisation; }
264
265 //remove this element from theElementTable
266 theElementTable[fIndexInTable] = 0;
267}
268
269//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
270
271void G4Element::ComputeDerivedQuantities()
272{
273 // some basic functions of the atomic number
274
275 // Store in table
276 theElementTable.push_back(this);
277 fIndexInTable = theElementTable.size() - 1;
278
279 // check if elements with same Z already exist
280 fIndexZ = 0;
281 for (size_t J=0 ; J<fIndexInTable ; J++) {
282 if (theElementTable[J]->GetZ() == fZeff) { ++fIndexZ; }
283 }
284 //nb of materials which use this element
285 fCountUse = 0;
286
287 // Radiation Length
288 ComputeCoulombFactor();
289 ComputeLradTsaiFactor();
290
291 // parameters for energy loss by ionisation
292 if (fIonisation) { delete fIonisation; }
293 fIonisation = new G4IonisParamElm(fZeff);
294}
295
296//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
297
298void G4Element::ComputeCoulombFactor()
299{
300 //
301 // Compute Coulomb correction factor (Phys Rev. D50 3-1 (1994) page 1254)
302
303 const G4double k1 = 0.0083 , k2 = 0.20206 ,k3 = 0.0020 , k4 = 0.0369 ;
304
305 G4double az2 = (fine_structure_const*fZeff)*(fine_structure_const*fZeff);
306 G4double az4 = az2 * az2;
307
308 fCoulomb = (k1*az4 + k2 + 1./(1.+az2))*az2 - (k3*az4 + k4)*az4;
309}
310
311//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
312
313void G4Element::ComputeLradTsaiFactor()
314{
315 //
316 // Compute Tsai's Expression for the Radiation Length
317 // (Phys Rev. D50 3-1 (1994) page 1254)
318
319 const G4double Lrad_light[] = {5.31 , 4.79 , 4.74 , 4.71} ;
320 const G4double Lprad_light[] = {6.144 , 5.621 , 5.805 , 5.924} ;
321
322 const G4double logZ3 = std::log(fZeff)/3.;
323
324 G4double Lrad, Lprad;
325 G4int iz = (G4int)(fZeff+0.5) - 1 ;
326 if (iz <= 3) { Lrad = Lrad_light[iz] ; Lprad = Lprad_light[iz] ; }
327 else { Lrad = std::log(184.15) - logZ3 ; Lprad = std::log(1194.) - 2*logZ3;}
328
329 fRadTsai = 4*alpha_rcl2*fZeff*(fZeff*(Lrad-fCoulomb) + Lprad);
330}
331
332//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
333
334void G4Element::AddNaturalIsotopes()
335{
336 G4int Z = G4lrint(fZeff);
339 G4int N0 = nist->GetNistFirstIsotopeN(Z);
340 fNumberOfIsotopes = 0;
341 for(G4int i=0; i<n; ++i) {
342 if(nist->GetIsotopeAbundance(Z, N0+i) > 0.0) { ++fNumberOfIsotopes; }
343 }
344 theIsotopeVector = new G4IsotopeVector(fNumberOfIsotopes,0);
345 fRelativeAbundanceVector = new G4double[fNumberOfIsotopes];
346 G4int idx = 0;
347 G4double xsum = 0.0;
348 for(G4int i=0; i<n; ++i) {
349 G4int N = N0 + i;
350 G4double x = nist->GetIsotopeAbundance(Z, N);
351 if(x > 0.0) {
352 std::ostringstream strm;
353 strm << fSymbol << N;
354 (*theIsotopeVector)[idx] = new G4Isotope(strm.str(),Z, N, 0.0, 0);
355 fRelativeAbundanceVector[idx] = x;
356 xsum += x;
357 ++idx;
358 }
359 }
360 if(xsum != 0.0 && xsum != 1.0) {
361 for(G4int i=0; i<idx; ++i) { fRelativeAbundanceVector[i] /= xsum; }
362 }
363}
364
365//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
366
368{
369 if (i<0 || i>=fNbOfAtomicShells) {
371 ed << "Invalid argument " << i << " in for G4Element " << fName
372 << " with Z= " << fZeff
373 << " and Nshells= " << fNbOfAtomicShells
374 << G4endl;
375 G4Exception("G4Element::GetAtomicShell()", "mat016", FatalException, ed);
376 return 0.0;
377 }
378 return fAtomicShells[i];
379}
380
381//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
382
384{
385 if (i<0 || i>=fNbOfAtomicShells) {
387 ed << "Invalid argument " << i << " for G4Element " << fName
388 << " with Z= " << fZeff
389 << " and Nshells= " << fNbOfAtomicShells
390 << G4endl;
391 G4Exception("G4Element::GetNbOfShellElectrons()", "mat016", FatalException, ed);
392 return 0;
393 }
394 return fNbOfShellElectrons[i];
395}
396
397//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
398
400{
401 return &theElementTable;
402}
403
404//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
405
407{
408 return theElementTable.size();
409}
410
411//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
412
414{
415 // search the element by its name
416 for (size_t J=0 ; J<theElementTable.size() ; J++)
417 {
418 if (theElementTable[J]->GetName() == elementName)
419 return theElementTable[J];
420 }
421
422 // the element does not exist in the table
423 if (warning) {
424 G4cout << "\n---> warning from G4Element::GetElement(). The element: "
425 << elementName << " does not exist in the table. Return NULL pointer."
426 << G4endl;
427 }
428 return 0;
429}
430
431//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
432
434{
435 InitializePointers();
436 *this = right;
437
438 // Store this new element in table and set the index
439 theElementTable.push_back(this);
440 fIndexInTable = theElementTable.size() - 1;
441}
442
443//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
444
445const G4Element& G4Element::operator=(const G4Element& right)
446{
447 if (this != &right)
448 {
449 fName = right.fName;
450 fSymbol = right.fSymbol;
451 fZeff = right.fZeff;
452 fNeff = right.fNeff;
453 fAeff = right.fAeff;
454
455 if (fAtomicShells) delete [] fAtomicShells;
456 fNbOfAtomicShells = right.fNbOfAtomicShells;
457 fAtomicShells = new G4double[fNbOfAtomicShells];
458
459 if (fNbOfShellElectrons) delete [] fNbOfShellElectrons;
460 fNbOfAtomicShells = right.fNbOfAtomicShells;
461 fNbOfShellElectrons = new G4int[fNbOfAtomicShells];
462
463 for ( G4int i = 0; i < fNbOfAtomicShells; i++ )
464 {
465 fAtomicShells[i] = right.fAtomicShells[i];
466 fNbOfShellElectrons[i] = right.fNbOfShellElectrons[i];
467 }
468 if (theIsotopeVector) delete theIsotopeVector;
469 if (fRelativeAbundanceVector) delete [] fRelativeAbundanceVector;
470
471 fNumberOfIsotopes = right.fNumberOfIsotopes;
472 if (fNumberOfIsotopes > 0)
473 {
474 theIsotopeVector = new G4IsotopeVector(fNumberOfIsotopes,0);
475 fRelativeAbundanceVector = new G4double[fNumberOfIsotopes];
476 for (size_t i=0;i<fNumberOfIsotopes;i++)
477 {
478 (*theIsotopeVector)[i] = (*right.theIsotopeVector)[i];
479 fRelativeAbundanceVector[i] = right.fRelativeAbundanceVector[i];
480 }
481 }
482 ComputeDerivedQuantities();
483 }
484 return *this;
485}
486
487//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
488
490{
491 return (this == (G4Element*) &right);
492}
493
494//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
495
497{
498 return (this != (G4Element*) &right);
499}
500
501//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
502
503std::ostream& operator<<(std::ostream& flux, G4Element* element)
504{
505 std::ios::fmtflags mode = flux.flags();
506 flux.setf(std::ios::fixed,std::ios::floatfield);
507 G4long prec = flux.precision(3);
508
509 flux
510 << " Element: " << element->fName << " (" << element->fSymbol << ")"
511 << " Z = " << std::setw(4) << std::setprecision(1) << element->fZeff
512 << " N = " << std::setw(5) << std::setprecision(1) << element->fNeff
513 << " A = " << std::setw(6) << std::setprecision(2)
514 << (element->fAeff)/(g/mole) << " g/mole";
515
516 for (size_t i=0; i<element->fNumberOfIsotopes; i++)
517 flux
518 << "\n ---> " << (*(element->theIsotopeVector))[i]
519 << " abundance: " << std::setw(6) << std::setprecision(2)
520 << (element->fRelativeAbundanceVector[i])/perCent << " %";
521
522 flux.precision(prec);
523 flux.setf(mode,std::ios::floatfield);
524 return flux;
525}
526
527//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
528
529 std::ostream& operator<<(std::ostream& flux, G4Element& element)
530{
531 flux << &element;
532 return flux;
533}
534
535//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
536
537std::ostream& operator<<(std::ostream& flux, G4ElementTable ElementTable)
538{
539 //Dump info for all known elements
540 flux << "\n***** Table : Nb of elements = " << ElementTable.size()
541 << " *****\n" << G4endl;
542
543 for (size_t i=0; i<ElementTable.size(); i++) flux << ElementTable[i]
544 << G4endl << G4endl;
545
546 return flux;
547}
548
549//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
std::vector< G4Element * > G4ElementTable
std::ostream & operator<<(std::ostream &flux, G4Element *element)
Definition: G4Element.cc:503
@ JustWarning
@ FatalException
std::vector< G4Isotope * > G4IsotopeVector
double G4double
Definition: G4Types.hh:64
long G4long
Definition: G4Types.hh:68
int G4int
Definition: G4Types.hh:66
bool G4bool
Definition: G4Types.hh:67
#define G4endl
Definition: G4ios.hh:52
G4DLLIMPORT std::ostream G4cout
static G4int GetNumberOfElectrons(G4int Z, G4int SubshellNb)
static G4double GetBindingEnergy(G4int Z, G4int SubshellNb)
static G4int GetNumberOfShells(G4int Z)
virtual ~G4Element()
Definition: G4Element.cc:255
G4double GetZ() const
Definition: G4Element.hh:131
static size_t GetNumberOfElements()
Definition: G4Element.cc:406
void AddIsotope(G4Isotope *isotope, G4double RelativeAbundance)
Definition: G4Element.cc:147
G4Element(const G4String &name, const G4String &symbol, G4double Zeff, G4double Aeff)
Definition: G4Element.cc:74
G4int operator==(const G4Element &) const
Definition: G4Element.cc:489
const G4String & GetName() const
Definition: G4Element.hh:127
static const G4ElementTable * GetElementTable()
Definition: G4Element.cc:399
G4int GetNbOfShellElectrons(G4int index) const
Definition: G4Element.cc:383
G4int operator!=(const G4Element &) const
Definition: G4Element.cc:496
G4double GetAtomicShell(G4int index) const
Definition: G4Element.cc:367
static G4Element * GetElement(G4String name, G4bool warning=true)
Definition: G4Element.cc:413
void increaseCountUse()
Definition: G4Isotope.hh:135
G4int GetZ() const
Definition: G4Isotope.hh:91
G4int GetNumberOfNistIsotopes(G4int Z) const
G4int GetNistFirstIsotopeN(G4int Z) const
static G4NistManager * Instance()
G4double GetIsotopeAbundance(G4int Z, G4int N) const
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *comments)
Definition: G4Exception.cc:41
std::ostringstream G4ExceptionDescription
Definition: globals.hh:76
int G4lrint(double ad)
Definition: templates.hh:163