91 if (density < universe_mean_density) {
92 G4cout <<
" G4Material WARNING:"
93 <<
" define a material with density=0 is not allowed. \n"
94 <<
" The material " << name <<
" will be constructed with the"
95 <<
" default minimal density: " << universe_mean_density / (g / cm3) <<
"g/cm3"
97 density = universe_mean_density;
103 fPressure = pressure;
107 fNbComponents = fNumberOfElements = 1;
114 if (elm ==
nullptr) {
115 elm =
new G4Element(
"ELM_" + name, name, z, a);
117 theElementVector->push_back(elm);
119 fMassFractionVector =
new G4double[1];
120 fMassFractionVector[0] = 1.;
121 fMassOfMolecule = a / CLHEP::Avogadro;
124 if (fDensity > kGasThreshold) {
132 ComputeDerivedQuantities();
144 InitializePointers();
146 if (density < universe_mean_density) {
147 G4cout <<
"--- Warning from G4Material::G4Material()"
148 <<
" define a material with density=0 is not allowed. \n"
149 <<
" The material " << name <<
" will be constructed with the"
150 <<
" default minimal density: " << universe_mean_density / (g / cm3) <<
"g/cm3"
152 density = universe_mean_density;
158 fPressure = pressure;
160 fNbComponents = nComponents;
161 fMassFraction =
true;
164 if (fDensity > kGasThreshold) {
181 InitializePointers();
183 if (density < universe_mean_density) {
184 G4cout <<
"--- Warning from G4Material::G4Material()"
185 <<
" define a material with density=0 is not allowed. \n"
186 <<
" The material " << name <<
" will be constructed with the"
187 <<
" default minimal density: " << universe_mean_density / (g / cm3) <<
"g/cm3"
189 density = universe_mean_density;
195 fPressure = pressure;
197 fBaseMaterial = bmat;
199 if (
nullptr != ptr) {
201 ptr = ptr->GetBaseMaterial();
202 if (
nullptr == ptr) {
210 fMassOfMolecule = fBaseMaterial->GetMassOfMolecule();
212 fNumberOfElements = (
G4int)fBaseMaterial->GetNumberOfElements();
213 fNbComponents = fNumberOfElements;
215 CopyPointersOfBaseMaterial();
222 if (fBaseMaterial ==
nullptr) {
223 delete theElementVector;
225 delete[] fMassFractionVector;
226 delete[] fAtomsVector;
229 delete[] fVecNbOfAtomsPerVolume;
238void G4Material::InitializePointers()
240 fBaseMaterial =
nullptr;
241 fMaterialPropertiesTable =
nullptr;
242 theElementVector =
nullptr;
243 fAtomsVector =
nullptr;
244 fMassFractionVector =
nullptr;
245 fVecNbOfAtomsPerVolume =
nullptr;
248 fSandiaTable =
nullptr;
250 fDensity = fFreeElecDensity = fTemp = fPressure = 0.0;
251 fTotNbOfAtomsPerVolume = 0.0;
252 fTotNbOfElectPerVolume = 0.0;
253 fRadlen = fNuclInterLen = fMassOfMolecule = 0.0;
256 fNumberOfElements = fNbComponents = fIdxComponent = 0;
257 fMassFraction =
true;
258 fChemicalFormula =
"";
262 for (std::size_t i = 0; i < fIndexInTable; ++i) {
264 G4cout <<
"G4Material WARNING: duplicate name of material " << fName <<
G4endl;
273void G4Material::ComputeDerivedQuantities()
279 fTotNbOfAtomsPerVolume = 0.;
280 delete[] fVecNbOfAtomsPerVolume;
281 fVecNbOfAtomsPerVolume =
new G4double[fNumberOfElements];
282 fTotNbOfElectPerVolume = 0.;
283 fFreeElecDensity = 0.;
284 const G4double elecTh = 15. * CLHEP::eV;
285 for (
G4int i = 0; i < fNumberOfElements; ++i) {
286 Zi = (*theElementVector)[i]->GetZ();
287 Ai = (*theElementVector)[i]->GetA();
288 fVecNbOfAtomsPerVolume[i] = Avogadro * fDensity * fMassFractionVector[i] / Ai;
289 fTotNbOfAtomsPerVolume += fVecNbOfAtomsPerVolume[i];
290 fTotNbOfElectPerVolume += fVecNbOfAtomsPerVolume[i] * Zi;
297 ComputeRadiationLength();
298 ComputeNuclearInterLength();
300 if (fIonisation ==
nullptr) {
301 fIonisation =
new G4IonisParamMat(
this);
303 if (fSandiaTable ==
nullptr) {
304 fSandiaTable =
new G4SandiaTable(
this);
310void G4Material::CopyPointersOfBaseMaterial()
312 G4double factor = fDensity / fBaseMaterial->GetDensity();
313 fTotNbOfAtomsPerVolume = factor * fBaseMaterial->GetTotNbOfAtomsPerVolume();
314 fTotNbOfElectPerVolume = factor * fBaseMaterial->GetTotNbOfElectPerVolume();
315 fFreeElecDensity = factor * fBaseMaterial->GetFreeElectronDensity();
318 fState = fBaseMaterial->GetState();
321 theElementVector =
const_cast<G4ElementVector*
>(fBaseMaterial->GetElementVector());
322 fMassFractionVector =
const_cast<G4double*
>(fBaseMaterial->GetFractionVector());
323 fAtomsVector =
const_cast<G4int*
>(fBaseMaterial->GetAtomsVector());
325 const G4double* v = fBaseMaterial->GetVecNbOfAtomsPerVolume();
326 delete[] fVecNbOfAtomsPerVolume;
327 fVecNbOfAtomsPerVolume =
new G4double[fNumberOfElements];
328 for (
G4int i = 0; i < fNumberOfElements; ++i) {
329 fVecNbOfAtomsPerVolume[i] = factor * v[i];
331 fRadlen = fBaseMaterial->GetRadlen() / factor;
332 fNuclInterLen = fBaseMaterial->GetNuclearInterLength() / factor;
334 if (fIonisation ==
nullptr) {
335 fIonisation =
new G4IonisParamMat(
this);
337 fIonisation->SetMeanExcitationEnergy(fBaseMaterial->GetIonisation()->GetMeanExcitationEnergy());
338 if (fBaseMaterial->GetIonisation()->GetDensityEffectCalculator() !=
nullptr) {
342 fSandiaTable = fBaseMaterial->GetSandiaTable();
343 fMaterialPropertiesTable = fBaseMaterial->GetMaterialPropertiesTable();
351 if (0 == fIdxComponent) {
352 fMassFraction =
false;
353 fAtoms =
new std::vector<G4int>;
354 fElm =
new std::vector<const G4Element*>;
356 if (fIdxComponent >= fNbComponents) {
358 ed <<
"For material " << fName <<
" and added element " << elm->
GetName()
359 <<
" with Natoms=" << nAtoms
360 <<
" wrong attempt to add more than the declared number of elements " << fIdxComponent
361 <<
" >= " << fNbComponents;
366 ed <<
"For material " << fName <<
" and added element " << elm->
GetName()
367 <<
" with Natoms=" << nAtoms <<
" problem: cannot add by number of atoms after "
368 <<
"addition of elements by mass fraction";
373 ed <<
"For material " << fName <<
" and added element " << elm->
GetName()
374 <<
" with Natoms=" << nAtoms <<
" problem: number of atoms should be above zero";
380 if (! fElm->empty()) {
381 for (
G4int i = 0; i < fNumberOfElements; ++i) {
382 if (elm == (*fElm)[i]) {
383 (*fAtoms)[i] += nAtoms;
390 fElm->push_back(elm);
391 fAtoms->push_back(nAtoms);
397 if (fIdxComponent == fNbComponents) {
399 theElementVector->reserve(fNumberOfElements);
400 fAtomsVector =
new G4int[fNumberOfElements];
401 fMassFractionVector =
new G4double[fNumberOfElements];
404 for (
G4int i = 0; i < fNumberOfElements; ++i) {
405 theElementVector->push_back((*fElm)[i]);
406 fAtomsVector[i] = (*fAtoms)[i];
407 G4double w = fAtomsVector[i] * (*fElm)[i]->GetA();
409 fMassFractionVector[i] = w;
411 for (
G4int i = 0; i < fNumberOfElements; ++i) {
412 fMassFractionVector[i] /= Amol;
416 fMassOfMolecule = Amol / CLHEP::Avogadro;
417 ComputeDerivedQuantities();
426 if (fraction < 0.0 || fraction > 1.0) {
428 ed <<
"For material " << fName <<
" and added element " << elm->
GetName()
429 <<
" massFraction= " << fraction <<
" is wrong ";
432 if (! fMassFraction) {
434 ed <<
"For material " << fName <<
" and added element " << elm->
GetName()
435 <<
", massFraction= " << fraction <<
", fIdxComponent=" << fIdxComponent
436 <<
" problem: cannot add by mass fraction after "
437 <<
"addition of elements by number of atoms";
440 if (fIdxComponent >= fNbComponents) {
442 ed <<
"For material " << fName <<
" and added element " << elm->
GetName()
443 <<
", massFraction= " << fraction <<
", fIdxComponent=" << fIdxComponent
444 <<
"; attempt to add more than the declared number of components " << fIdxComponent
445 <<
" >= " << fNbComponents;
448 if (0 == fIdxComponent) {
449 fElmFrac =
new std::vector<G4double>;
450 fElm =
new std::vector<const G4Element*>;
455 if (! fElm->empty()) {
456 for (
G4int i = 0; i < fNumberOfElements; ++i) {
457 if (elm == (*fElm)[i]) {
458 (*fElmFrac)[i] += fraction;
465 fElm->push_back(elm);
466 fElmFrac->push_back(fraction);
472 if (fIdxComponent == fNbComponents) {
482 if (fraction < 0.0 || fraction > 1.0) {
484 ed <<
"For material " << fName <<
" and added material " << material->
GetName()
485 <<
", massFraction= " << fraction <<
" is wrong ";
488 if (! fMassFraction) {
490 ed <<
"For material " << fName <<
" and added material " << material->
GetName()
491 <<
", massFraction= " << fraction <<
", fIdxComponent=" << fIdxComponent
492 <<
" problem: cannot add by mass fraction after "
493 <<
"addition of elements by number of atoms";
496 if (fIdxComponent >= fNbComponents) {
498 ed <<
"For material " << fName <<
" and added material " << material->
GetName()
499 <<
", massFraction= " << fraction
500 <<
"; attempt to add more than the declared number of components " << fIdxComponent
501 <<
" >= " << fNbComponents;
504 if (0 == fIdxComponent) {
505 fElmFrac =
new std::vector<G4double>;
506 fElm =
new std::vector<const G4Element*>;
511 for (
G4int j = 0; j < nelm; ++j) {
515 if (! fElm->empty()) {
516 for (
G4int i = 0; i < fNumberOfElements; ++i) {
517 if (elm == (*fElm)[i]) {
518 (*fElmFrac)[i] += fraction * frac[j];
525 fElm->push_back(elm);
526 fElmFrac->push_back(fraction * frac[j]);
531 fMatComponents[material] = fraction;
535 if (fIdxComponent == fNbComponents) {
542void G4Material::FillVectors()
546 theElementVector->reserve(fNumberOfElements);
547 fAtomsVector =
new G4int[fNumberOfElements];
548 fMassFractionVector =
new G4double[fNumberOfElements];
551 for (
G4int i = 0; i < fNumberOfElements; ++i) {
552 theElementVector->push_back((*fElm)[i]);
553 fMassFractionVector[i] = (*fElmFrac)[i];
554 wtSum += fMassFractionVector[i];
560 if (std::abs(1. - wtSum) > perThousand) {
562 ed <<
"For material " << fName <<
" sum of fractional masses " << wtSum
563 <<
" is not 1 - results may be wrong";
568 for (
G4int i = 0; i < fNumberOfElements; ++i) {
569 fMassFractionVector[i] *=
coeff;
570 Amol += fMassFractionVector[i] * (*theElementVector)[i]->GetA();
572 for (
G4int i = 0; i < fNumberOfElements; ++i) {
573 fAtomsVector[i] =
G4lrint(fMassFractionVector[i] * Amol / (*theElementVector)[i]->
GetA());
575 ComputeDerivedQuantities();
580void G4Material::ComputeRadiationLength()
583 for (
G4int i = 0; i < fNumberOfElements; ++i) {
584 radinv += fVecNbOfAtomsPerVolume[i] * ((*theElementVector)[i]->GetfRadTsai());
586 fRadlen = (radinv <= 0.0 ?
DBL_MAX : 1. / radinv);
591void G4Material::ComputeNuclearInterLength()
593 const G4double lambda1 = CLHEP::amu*CLHEP::cm2 / (35.0 * CLHEP::g);
595 for (
G4int i = 0; i < fNumberOfElements; ++i) {
596 G4int Z = (*theElementVector)[i]->GetZasInt();
597 G4double A = (*theElementVector)[i]->GetN();
599 NILinv += fVecNbOfAtomsPerVolume[i] *
A;
606 fNuclInterLen = 1.e+20*CLHEP::m;
607 if (fNuclInterLen * NILinv > 1.0) {
608 fNuclInterLen = 1.0 / NILinv;
617 fChemicalFormula = chF;
625 if (val >= 0. && ! IsLocked()) {
626 fFreeElecDensity = val;
635 if (
nullptr == fIonisation) {
638 fIonisation->ComputeDensityEffectOnFly(val);
653 static Holder _holder;
654 return &_holder.instance;
667 if (j->GetName() == materialName) {
674 G4cout <<
"G4Material::GetMaterial() WARNING: The material: " << materialName
675 <<
" does not exist in the table. Return NULL pointer." <<
G4endl;
686 if (1 == mat->GetNumberOfElements() && z == mat->GetZ() && a == mat->GetA() &&
687 dens == mat->GetDensity())
701 if (nComp == mat->GetNumberOfElements() && dens == mat->GetDensity()) {
712 if (fNumberOfElements > 1) {
714 ed <<
"For material " << fName <<
" ERROR in GetZ() - Nelm=" << fNumberOfElements
715 <<
" > 1, which is not allowed";
718 return (*theElementVector)[0]->GetZ();
725 if (fNumberOfElements > 1) {
727 ed <<
"For material " << fName <<
" ERROR in GetA() - Nelm=" << fNumberOfElements
728 <<
" > 1, which is not allowed";
731 return (*theElementVector)[0]->GetA();
738 std::ios::fmtflags mode = flux.flags();
739 flux.setf(std::ios::fixed, std::ios::floatfield);
740 G4long prec = flux.precision(3);
742 flux <<
" Material: " << std::setw(8) << material->fName <<
" " << material->fChemicalFormula
744 <<
" density: " << std::setw(6) << std::setprecision(3)
745 <<
G4BestUnit(material->fDensity,
"Volumic Mass") <<
" RadL: " << std::setw(7)
746 << std::setprecision(3) <<
G4BestUnit(material->fRadlen,
"Length")
747 <<
" Nucl.Int.Length: " << std::setw(7) << std::setprecision(3)
748 <<
G4BestUnit(material->fNuclInterLen,
"Length") <<
"\n"
749 << std::setw(30) <<
" Imean: " << std::setw(7) << std::setprecision(3)
751 <<
" temperature: " << std::setw(6) << std::setprecision(2)
752 << (material->fTemp) / CLHEP::kelvin <<
" K"
753 <<
" pressure: " << std::setw(6) << std::setprecision(2)
754 << (material->fPressure) / CLHEP::atmosphere <<
" atm"
757 for (
G4int i = 0; i < material->fNumberOfElements; i++) {
758 flux <<
"\n ---> " << (*(material->theElementVector))[i]
759 <<
"\n ElmMassFraction: " << std::setw(6) << std::setprecision(2)
760 << (material->fMassFractionVector[i]) / perCent <<
" %"
761 <<
" ElmAbundance " << std::setw(6) << std::setprecision(2)
762 << 100 * (material->fVecNbOfAtomsPerVolume[i]) / (material->fTotNbOfAtomsPerVolume)
765 flux.precision(prec);
766 flux.setf(mode, std::ios::floatfield);
788 flux <<
"\n***** Table : Nb of materials = " << MaterialTable.size() <<
" *****\n" <<
G4endl;
790 for (
auto i : MaterialTable) {
805 if (fMaterialPropertiesTable != anMPT && ! IsLocked()) {
806 delete fMaterialPropertiesTable;
807 fMaterialPropertiesTable = anMPT;
813G4bool G4Material::IsLocked()
std::vector< const G4Element * > G4ElementVector
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
std::ostringstream G4ExceptionDescription
std::vector< G4Material * > G4MaterialTable
std::ostream & operator<<(std::ostream &flux, const G4Material *material)
G4TemplateRNGHelper< G4long > * G4TemplateRNGHelper< G4long >::instance
G4GLOB_DLL std::ostream G4cout
static G4int GetNumberOfFreeElectrons(G4int Z, G4double th)
const G4String & GetName() const
G4double GetMeanExcitationEnergy() const
void ComputeDensityEffectOnFly(G4bool val)
const G4String & GetChemicalFormula() const
const G4Element * GetElement(G4int iel) const
virtual G4bool IsExtended() const
const G4double * GetFractionVector() const
G4IonisParamMat * GetIonisation() const
void SetChemicalFormula(const G4String &chF)
void AddElementByNumberOfAtoms(const G4Element *elm, G4int nAtoms)
static std::size_t GetNumberOfMaterials()
G4Material(const G4String &name, G4double z, G4double a, G4double density, G4State state=kStateUndefined, G4double temp=NTP_Temperature, G4double pressure=CLHEP::STP_Pressure)
static G4MaterialTable * GetMaterialTable()
void AddMaterial(G4Material *material, G4double fraction)
std::size_t GetNumberOfElements() const
const G4String & GetName() const
void SetMaterialPropertiesTable(G4MaterialPropertiesTable *anMPT)
void SetFreeElectronDensity(G4double val)
void AddElementByMassFraction(const G4Element *elm, G4double fraction)
static G4Material * GetMaterial(const G4String &name, G4bool warning=true)
static G4NistManager * Instance()
G4Element * FindOrBuildElement(G4int Z, G4bool isotopes=true)
static G4Pow * GetInstance()
G4double A23(G4double A) const
const G4ApplicationState & GetCurrentState() const
static G4StateManager * GetStateManager()