Geant4 11.3.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// 26-06-96: Code uses operators (+=, *=, ++, -> etc.) correctly, P. Urban
27// 09-07-96: new data members added by L.Urban
28// 17-01-97: aesthetic rearrangement, M.Maire
29// 20-01-97: Compute Tsai's formula for the rad length, M.Maire
30// 21-01-97: remove mixture flag, M.Maire
31// 24-01-97: ComputeIonisationParameters().
32// new data member: fTaul, M.Maire
33// 29-01-97: Forbidden to create Element with Z<1 or N<Z, M.Maire
34// 20-03-97: corrected initialization of pointers, M.Maire
35// 28-04-98: atomic subshell binding energies stuff, V. Grichine
36// 09-07-98: Ionisation parameters removed from the class, M.Maire
37// 16-11-98: name Subshell -> Shell; GetBindingEnergy() (mma)
38// 09-03-01: assignement operator revised (mma)
39// 02-05-01: check identical Z in AddIsotope (marc)
40// 03-05-01: flux.precision(prec) at begin/end of operator<<
41// 13-09-01: suppression of the data member fIndexInTable
42// 14-09-01: fCountUse: nb of materials which use this element
43// 26-02-02: fIndexInTable renewed
44// 30-03-05: warning in GetElement(elementName)
45// 15-11-05: GetElement(elementName, G4bool warning=true)
46// 17-10-06: Add fNaturalAbundances (V.Ivanchenko)
47// 27-07-07: improve destructor (V.Ivanchenko)
48// 18-10-07: move definition of material index to ComputeDerivedQuantities (VI)
49// 25.10.11: new scheme for G4Exception (mma)
50// 05-03-12: always create isotope vector (V.Ivanchenko)
51
52#include "G4Element.hh"
53
54#include "G4AtomicShells.hh"
55#include "G4Log.hh"
56#include "G4NistManager.hh"
58#include "G4SystemOfUnits.hh"
59
60#include <iomanip>
61#include <sstream>
62#include <utility>
63
64// Constructor to Generate an element from scratch
65
66G4Element::G4Element(const G4String& name, const G4String& symbol, G4double zeff, G4double aeff)
67 : fName(name), fSymbol(symbol)
68{
69 G4int iz = G4lrint(zeff);
70 if (iz < 1) {
72 ed << "Failed to create G4Element " << name << " Z= " << zeff << " < 1 !";
73 G4Exception("G4Element::G4Element()", "mat011", FatalException, ed);
74 }
75 if (std::abs(zeff - iz) > perMillion) {
77 ed << "G4Element Warning: " << name << " Z= " << zeff << " A= " << aeff / (g / mole);
78 G4Exception("G4Element::G4Element()", "mat017", JustWarning, ed);
79 }
80
81 InitializePointers();
82
83 fZeff = zeff;
84 fAeff = aeff;
85 fNeff = fAeff / (g / mole);
86
87 if (fNeff < 1.0) {
88 fNeff = 1.0;
89 }
90
91 if (fNeff < zeff) {
93 ed << "Failed to create G4Element " << name << " with Z= " << zeff << " N= " << fNeff
94 << " N < Z is not allowed" << G4endl;
95 G4Exception("G4Element::G4Element()", "mat012", FatalException, ed);
96 }
97
98 fNbOfAtomicShells = G4AtomicShells::GetNumberOfShells(iz);
99 fAtomicShells = new G4double[fNbOfAtomicShells];
100 fNbOfShellElectrons = new G4int[fNbOfAtomicShells];
101
102 AddNaturalIsotopes();
103
104 for (G4int i = 0; i < fNbOfAtomicShells; ++i) {
105 fAtomicShells[i] = G4AtomicShells::GetBindingEnergy(iz, i);
106 fNbOfShellElectrons[i] = G4AtomicShells::GetNumberOfElectrons(iz, i);
107 }
108 ComputeDerivedQuantities();
109}
110
111//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
112
113// Constructor to Generate element from a List of 'nIsotopes' isotopes, added
114// via AddIsotope
115
116G4Element::G4Element(const G4String& name, const G4String& symbol, G4int nIsotopes)
117 : fName(name), fSymbol(symbol)
118{
119 InitializePointers();
120
121 if (0 >= nIsotopes) {
123 ed << "Failed to create G4Element " << name << " <" << symbol << "> with " << nIsotopes
124 << " isotopes.";
125 G4Exception("G4Element::G4Element()", "mat012", FatalException, ed);
126 }
127 else {
128 auto n = (std::size_t)nIsotopes;
129 theIsotopeVector = new G4IsotopeVector(n, nullptr);
130 fRelativeAbundanceVector = new G4double[nIsotopes];
131 }
132}
133
134//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
135
136// Add an isotope to the element
137
138void G4Element::AddIsotope(G4Isotope* isotope, G4double abundance)
139{
140 if (theIsotopeVector == nullptr) {
142 ed << "Failed to add Isotope to G4Element " << fName << " with Z= " << fZeff
143 << " N= " << fNeff;
144 G4Exception("G4Element::AddIsotope()", "mat013", FatalException, ed);
145 return;
146 }
147 G4int iz = isotope->GetZ();
148
149 // filling ...
150 if (fNumberOfIsotopes < (G4int)theIsotopeVector->size()) {
151 // check same Z
152 if (fNumberOfIsotopes == 0) {
153 fZeff = G4double(iz);
154 }
155 else if (G4double(iz) != fZeff) {
157 ed << "Failed to add Isotope Z= " << iz << " to G4Element " << fName
158 << " with different Z= " << fZeff << fNeff;
159 G4Exception("G4Element::AddIsotope()", "mat014", FatalException, ed);
160 return;
161 }
162 // Z ok
163 fRelativeAbundanceVector[fNumberOfIsotopes] = abundance;
164 (*theIsotopeVector)[fNumberOfIsotopes] = isotope;
165 ++fNumberOfIsotopes;
166 }
167 else {
169 ed << "Failed to add Isotope Z= " << iz << " to G4Element " << fName
170 << " - more isotopes than declared.";
171 G4Exception("G4Element::AddIsotope()", "mat015", FatalException, ed);
172 return;
173 }
174
175 // filled.
176 if (fNumberOfIsotopes == (G4int)theIsotopeVector->size()) {
177 G4double wtSum = 0.0;
178 fAeff = 0.0;
179 for (G4int i = 0; i < fNumberOfIsotopes; ++i) {
180 fAeff += fRelativeAbundanceVector[i] * (*theIsotopeVector)[i]->GetA();
181 wtSum += fRelativeAbundanceVector[i];
182 }
183 if (wtSum > 0.0) {
184 fAeff /= wtSum;
185 }
186 fNeff = fAeff / (g / mole);
187
188 if (wtSum != 1.0) {
189 for (G4int i = 0; i < fNumberOfIsotopes; ++i) {
190 fRelativeAbundanceVector[i] /= wtSum;
191 }
192 }
193
194 fNbOfAtomicShells = G4AtomicShells::GetNumberOfShells(iz);
195 fAtomicShells = new G4double[fNbOfAtomicShells];
196 fNbOfShellElectrons = new G4int[fNbOfAtomicShells];
197
198 for (G4int j = 0; j < fNbOfAtomicShells; ++j) {
199 fAtomicShells[j] = G4AtomicShells::GetBindingEnergy(iz, j);
200 fNbOfShellElectrons[j] = G4AtomicShells::GetNumberOfElectrons(iz, j);
201 }
202 ComputeDerivedQuantities();
203 }
204}
205
206//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
207
208void G4Element::InitializePointers()
209{
210 theIsotopeVector = nullptr;
211 fRelativeAbundanceVector = nullptr;
212 fAtomicShells = nullptr;
213 fNbOfShellElectrons = nullptr;
214 fIonisation = nullptr;
215 fNumberOfIsotopes = 0;
216 fNaturalAbundance = false;
217
218 // add initialisation of all remaining members
219 fZeff = 0;
220 fNeff = 0;
221 fAeff = 0;
222 fNbOfAtomicShells = 0;
223 fIndexInTable = 0;
224 fCoulomb = 0.0;
225 fRadTsai = 0.0;
226 fZ = 0;
227}
228
229//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
230
232{
233 delete theIsotopeVector;
234 delete[] fRelativeAbundanceVector;
235 delete[] fAtomicShells;
236 delete[] fNbOfShellElectrons;
237 delete fIonisation;
238
239 // remove this element from the ElementTable
240 GetElementTableRef()[fIndexInTable] = nullptr;
241}
242
243//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
244
245void G4Element::ComputeDerivedQuantities()
246{
247 // some basic functions of the atomic number
248
249 // Store in table
250 GetElementTableRef().push_back(this);
251 fIndexInTable = GetElementTableRef().size() - 1;
252
253 // Radiation Length
254 ComputeCoulombFactor();
255 ComputeLradTsaiFactor();
256
257 // parameters for energy loss by ionisation
258 delete fIonisation;
259 fIonisation = new G4IonisParamElm(fZeff);
260 fZ = G4lrint(fZeff);
261}
262
263//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
264
265void G4Element::ComputeCoulombFactor()
266{
267 //
268 // Compute Coulomb correction factor (Phys Rev. D50 3-1 (1994) page 1254)
269
270 static const G4double k1 = 0.0083, k2 = 0.20206, k3 = 0.0020, k4 = 0.0369;
271
272 G4double az2 = (fine_structure_const * fZeff) * (fine_structure_const * fZeff);
273 G4double az4 = az2 * az2;
274
275 fCoulomb = (k1 * az4 + k2 + 1. / (1. + az2)) * az2 - (k3 * az4 + k4) * az4;
276}
277
278//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
279
280void G4Element::ComputeLradTsaiFactor()
281{
282 //
283 // Compute Tsai's Expression for the Radiation Length
284 // (Phys Rev. D50 3-1 (1994) page 1254)
285
286 static const G4double Lrad_light[] = {5.31, 4.79, 4.74, 4.71};
287 static const G4double Lprad_light[] = {6.144, 5.621, 5.805, 5.924};
288
289 const G4double logZ3 = G4Log(fZeff) / 3.;
290
291 G4double Lrad, Lprad;
292 G4int iz = G4lrint(fZeff) - 1;
293 static const G4double log184 = G4Log(184.15);
294 static const G4double log1194 = G4Log(1194.);
295 if (iz <= 3) {
296 Lrad = Lrad_light[iz];
297 Lprad = Lprad_light[iz];
298 }
299 else {
300 Lrad = log184 - logZ3;
301 Lprad = log1194 - 2 * logZ3;
302 }
303
304 fRadTsai = 4 * alpha_rcl2 * fZeff * (fZeff * (Lrad - fCoulomb) + Lprad);
305}
306
307//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
308
309void G4Element::AddNaturalIsotopes()
310{
311 G4int Z = G4lrint(fZeff);
312 G4NistManager* nist = G4NistManager::Instance();
314 G4int N0 = nist->GetNistFirstIsotopeN(Z);
315
316 if (fSymbol.empty()) {
317 const std::vector<G4String>& elmnames = G4NistManager::Instance()->GetNistElementNames();
318 if (Z < (G4int)elmnames.size()) {
319 fSymbol = elmnames[Z];
320 }
321 else {
322 fSymbol = fName;
323 }
324 }
325
326 fNumberOfIsotopes = 0;
327 for (G4int i = 0; i < n; ++i) {
328 if (nist->GetIsotopeAbundance(Z, N0 + i) > 0.0) {
329 ++fNumberOfIsotopes;
330 }
331 }
332 theIsotopeVector = new G4IsotopeVector((std::size_t)fNumberOfIsotopes, nullptr);
333 fRelativeAbundanceVector = new G4double[fNumberOfIsotopes];
334 G4int idx = 0;
335 G4double xsum = 0.0;
336 for (G4int i = 0; i < n; ++i) {
337 G4int N = N0 + i;
338 G4double x = nist->GetIsotopeAbundance(Z, N);
339 if (x > 0.0) {
340 std::ostringstream strm;
341 strm << fSymbol << N;
342 (*theIsotopeVector)[idx] = new G4Isotope(strm.str(), Z, N, 0.0, 0);
343 fRelativeAbundanceVector[idx] = x;
344 xsum += x;
345 ++idx;
346 }
347 }
348 if (xsum != 0.0 && xsum != 1.0) {
349 for (G4int i = 0; i < idx; ++i) {
350 fRelativeAbundanceVector[i] /= xsum;
351 }
352 }
353 fNaturalAbundance = true;
354}
355
356//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
357
359{
360 if (i < 0 || i >= fNbOfAtomicShells) {
362 ed << "Invalid argument " << i << " in for G4Element " << fName << " with Z= " << fZeff
363 << " and Nshells= " << fNbOfAtomicShells;
364 G4Exception("G4Element::GetAtomicShell()", "mat016", FatalException, ed);
365 return 0.0;
366 }
367 return fAtomicShells[i];
368}
369
370//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
371
373{
374 if (i < 0 || i >= fNbOfAtomicShells) {
376 ed << "Invalid argument " << i << " for G4Element " << fName << " with Z= " << fZeff
377 << " and Nshells= " << fNbOfAtomicShells;
378 G4Exception("G4Element::GetNbOfShellElectrons()", "mat016", FatalException, ed);
379 return 0;
380 }
381 return fNbOfShellElectrons[i];
382}
383
384//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
385
386G4ElementTable& G4Element::GetElementTableRef()
387{
388 struct Holder {
390 ~Holder() {
391 for(auto item : instance)
392 delete item;
393 }
394 };
395 static Holder _holder;
396 return _holder.instance;
397}
398
399//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
400
402{
403 return &GetElementTableRef();
404}
405
406//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
407
408size_t G4Element::GetNumberOfElements() { return GetElementTableRef().size(); }
409
410//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
411
412G4Element* G4Element::GetElement(const G4String& elementName, G4bool warning)
413{
414 // search the element by its name
415 for (auto const & J : GetElementTableRef())
416 {
417 if(J->GetName() == elementName)
418 {
419 return J;
420 }
421 }
422
423 // the element does not exist in the table
424 if (warning) {
425 G4cout << "\n---> warning from G4Element::GetElement(). The element: " << elementName
426 << " does not exist in the table. Return NULL pointer." << G4endl;
427 }
428 return nullptr;
429}
430
431//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
432
433std::ostream& operator<<(std::ostream& flux, const G4Element* element)
434{
435 std::ios::fmtflags mode = flux.flags();
436 flux.setf(std::ios::fixed, std::ios::floatfield);
437 G4long prec = flux.precision(3);
438
439 flux << " Element: " << element->fName << " (" << element->fSymbol << ")"
440 << " Z = " << std::setw(4) << std::setprecision(1) << element->fZeff
441 << " N = " << std::setw(5) << std::setprecision(1) << G4lrint(element->fNeff)
442 << " A = " << std::setw(6) << std::setprecision(3) << (element->fAeff) / (g / mole)
443 << " g/mole";
444
445 for (G4int i = 0; i < element->fNumberOfIsotopes; i++) {
446 flux << "\n ---> " << (*(element->theIsotopeVector))[i]
447 << " abundance: " << std::setw(6) << std::setprecision(3)
448 << (element->fRelativeAbundanceVector[i]) / perCent << " %";
449 }
450
451 flux.precision(prec);
452 flux.setf(mode, std::ios::floatfield);
453 return flux;
454}
455
456//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
457
458std::ostream& operator<<(std::ostream& flux, const G4Element& element)
459{
460 flux << &element;
461 return flux;
462}
463
464//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
465
466std::ostream& operator<<(std::ostream& flux, const G4ElementTable& ElementTable)
467{
468 // Dump info for all known elements
469 flux << "\n***** Table : Nb of elements = " << ElementTable.size() << " *****\n" << G4endl;
470
471 for (auto const & i : ElementTable) {
472 flux << i << G4endl << G4endl;
473 }
474
475 return flux;
476}
477
478//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
479
480std::ostream& operator<<(std::ostream& flux, const G4ElementVector& ElementVector)
481{
482 // Dump info for all known elements
483 flux << "\n***** Vector : Nb of elements = " << ElementVector.size() << " *****\n" << G4endl;
484
485 for (auto const & i : ElementVector) {
486 flux << i << G4endl << G4endl;
487 }
488
489 return flux;
490}
std::vector< G4Element * > G4ElementTable
std::vector< const G4Element * > G4ElementVector
std::ostream & operator<<(std::ostream &flux, const G4Element *element)
Definition G4Element.cc:433
@ fIonisation
@ JustWarning
@ FatalException
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
std::ostringstream G4ExceptionDescription
std::vector< G4Isotope * > G4IsotopeVector
G4double G4Log(G4double x)
Definition G4Log.hh:227
G4TemplateRNGHelper< G4long > * G4TemplateRNGHelper< G4long >::instance
double G4double
Definition G4Types.hh:83
long G4long
Definition G4Types.hh:87
bool G4bool
Definition G4Types.hh:86
int G4int
Definition G4Types.hh:85
#define G4endl
Definition G4ios.hh:67
G4GLOB_DLL 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:231
static std::size_t GetNumberOfElements()
Definition G4Element.cc:408
void AddIsotope(G4Isotope *isotope, G4double RelativeAbundance)
Definition G4Element.cc:138
G4Element(const G4String &name, const G4String &symbol, G4double Zeff, G4double Aeff)
Definition G4Element.cc:66
static const G4ElementTable * GetElementTable()
Definition G4Element.cc:401
G4int GetNbOfShellElectrons(G4int index) const
Definition G4Element.cc:372
static G4Element * GetElement(const G4String &name, G4bool warning=true)
Definition G4Element.cc:412
G4double GetAtomicShell(G4int index) const
Definition G4Element.cc:358
G4int GetZ() const
Definition G4Isotope.hh:80
G4int GetNumberOfNistIsotopes(G4int Z) const
const std::vector< G4String > & GetNistElementNames() const
G4int GetNistFirstIsotopeN(G4int Z) const
static G4NistManager * Instance()
G4double GetIsotopeAbundance(G4int Z, G4int N) const
#define N
Definition crc32.c:57
int G4lrint(double ad)
Definition templates.hh:134