2022-12-30 17:29:07 +01:00
|
|
|
|
|
|
|
[![Arduino CI](https://github.com/RobTillaart/AtomicWeight/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
|
|
|
|
[![Arduino-lint](https://github.com/RobTillaart/AtomicWeight/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/AtomicWeight/actions/workflows/arduino-lint.yml)
|
|
|
|
[![JSON check](https://github.com/RobTillaart/AtomicWeight/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/AtomicWeight/actions/workflows/jsoncheck.yml)
|
|
|
|
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/AtomicWeight/blob/master/LICENSE)
|
|
|
|
[![GitHub release](https://img.shields.io/github/release/RobTillaart/AtomicWeight.svg?maxAge=3600)](https://github.com/RobTillaart/AtomicWeight/releases)
|
|
|
|
|
|
|
|
|
|
|
|
# AtomicWeight
|
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
Arduino library for atomic weights, and related functions.
|
2022-12-30 17:29:07 +01:00
|
|
|
|
|
|
|
|
|
|
|
## Description
|
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
This library is mainly written to be used for educational purposes.
|
2023-04-15 20:15:18 +02:00
|
|
|
Of course are other applications possible.
|
2023-04-14 13:34:00 +02:00
|
|
|
Learning the **periodic table of elements**, the abbreviations and weights.
|
2023-04-15 20:15:18 +02:00
|
|
|
It provides the number of electrons, neutrons and protons per element.
|
2022-12-31 17:38:51 +01:00
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
Furthermore the library has a **weight()** function, which returns the weight
|
2023-04-15 20:15:18 +02:00
|
|
|
of either an element or of a formula (e.g. a molecule) in **Daltons**.
|
|
|
|
One **Dalton** = 1/12th of the weight of a Carbon atom.
|
|
|
|
The SI abbreviation for **Dalton** == **Da**.
|
|
|
|
One might think of it as (roughly) the number of nucleons (= protons + neutrons).
|
|
|
|
|
|
|
|
The **weight()** function can also be used to get the weight of a particular element
|
2023-04-14 13:34:00 +02:00
|
|
|
within a formula, e.g. the total weight of all Oxygen atoms in a **H2SO4** molecule.
|
2023-04-12 16:52:57 +02:00
|
|
|
This latter function allows the library to calculate the **massPercentage()** too.
|
2022-12-31 17:38:51 +01:00
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
The library has a **count()** function, to count the atoms in a given formula.
|
|
|
|
Derived is the **atomPercentage()** function which returns the percentage of atoms
|
2023-04-15 20:15:18 +02:00
|
|
|
that is of a certain element. Oxygen in **H2S04** is 4 in 7 which is about 57.14%.
|
2023-04-14 13:34:00 +02:00
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
The library supports conversion from moles to grams and back.
|
2023-04-14 13:34:00 +02:00
|
|
|
This allows one to easily get the amount of grams of some formula one has to weigh
|
|
|
|
to get a needed amount of moles.
|
|
|
|
With these functions, in combination with a load cell, one could create a "molar-scale".
|
|
|
|
Another application for the conversion functions is create lookup-tables, see example.
|
2022-12-30 17:29:07 +01:00
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
Note: the library is experimental. More testing is needed. Feedback welcome.
|
2022-12-30 17:29:07 +01:00
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
|
|
|
|
#### Internal
|
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
The PTOE class uses a table with "scaled" weights to save RAM.
|
|
|
|
- it stores weights as **uint16_t**, instead of floats (saves 236 bytes).
|
2023-04-14 13:34:00 +02:00
|
|
|
- the table (and thus the class) does not handle isotopes of elements.
|
2023-04-15 20:15:18 +02:00
|
|
|
- the scaling factor is named **ATOMIC_WEIGHT_FACTOR**
|
|
|
|
|
|
|
|
**Version 0.2.x**
|
|
|
|
- ATOMIC_WEIGHT_FACTOR = 1.0 / 201.3868 (found by searching - see example).
|
|
|
|
- relative error per element is less than 0.09% (~40% better than previous 0.15%).
|
|
|
|
- **Breaking:** numbers are **not** compatible with 0.1.x version
|
|
|
|
|
|
|
|
|
|
|
|
**Version 0.1.x (for reference)**
|
|
|
|
- ATOMIC_WEIGHT_FACTOR = 1.0 / 222.909 == weight heaviest element(118) / 65535.
|
|
|
|
- relative error per element is less than 0.15%.
|
2022-12-30 17:29:07 +01:00
|
|
|
|
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
#### Related
|
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
List of formulae to play with.
|
2023-04-12 16:52:57 +02:00
|
|
|
- https://en.wikipedia.org/wiki/Glossary_of_chemical_formulae
|
2023-04-15 20:15:18 +02:00
|
|
|
- https://en.wikipedia.org/wiki/Avogadro_constant
|
2023-04-12 16:52:57 +02:00
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
Libraries useful to build the "molar-scale"
|
2023-04-15 20:15:18 +02:00
|
|
|
- https://github.com/RobTillaart/HX711 (load cell)
|
|
|
|
- https://github.com/RobTillaart/weight (conversion)
|
|
|
|
- https://github.com/RobTillaart/printHelpers (scientific notation of numbers)
|
2023-04-14 13:34:00 +02:00
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
|
2022-12-30 17:29:07 +01:00
|
|
|
## Interface
|
|
|
|
|
|
|
|
```cpp
|
|
|
|
#include "AtomicWeight.h"
|
|
|
|
```
|
|
|
|
|
2022-12-31 18:30:42 +01:00
|
|
|
The parameter **element** in the following functions is 0..118.
|
2023-04-14 13:34:00 +02:00
|
|
|
(element 0 being 'n' == a single neutron).
|
2022-12-31 18:30:42 +01:00
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
- **PTOE(uint8_t size = 118)** Constructor (Periodic Table Of Elements).
|
|
|
|
Default it holds all 118 elements.
|
|
|
|
The parameter size is used in the **find()** and guards some parameters.
|
2022-12-30 17:29:07 +01:00
|
|
|
- **uint8_t electrons(uint8_t element)** returns the number of electrons of the element.
|
|
|
|
- **uint8_t neutrons(uint8_t element)** returns the number of neutrons of the element.
|
|
|
|
- **uint8_t protons(uint8_t element)** returns the number of protons of the element.
|
2022-12-31 17:38:51 +01:00
|
|
|
- **float weight(uint8_t element)** returns the weight of the element.
|
2023-04-15 20:15:18 +02:00
|
|
|
The error is less than 0.09%, as the internal table uses "compression" to save RAM.
|
2023-01-02 13:02:36 +01:00
|
|
|
- **float weight(char \* formula, char \* abbreviation == NULL)** see below.
|
2023-04-14 13:34:00 +02:00
|
|
|
- if (abbreviation == NULL) returns the weight of the whole formula.
|
|
|
|
- If (abbreviation != NULL) returns the total weight of one element in a formula.
|
|
|
|
- Returns 0 if it cannot parse the given formula.
|
2023-01-02 13:02:36 +01:00
|
|
|
- **float massPercentage(char \* formula, char \* abbreviation)**
|
2023-04-14 13:34:00 +02:00
|
|
|
Returns mass percentage of a given element in a formula.
|
2022-12-30 17:29:07 +01:00
|
|
|
- **uint8_t find(char \* abbreviation)** returns the element number.
|
2023-04-12 16:52:57 +02:00
|
|
|
This function is relative expensive as it searches linear through the internal array of elements.
|
2023-04-14 13:34:00 +02:00
|
|
|
Note: the find function is case sensitive.
|
2022-12-30 17:29:07 +01:00
|
|
|
- **char \* name(uint8_t element)** returns the abbreviation of element.
|
2023-04-14 13:34:00 +02:00
|
|
|
If the element is out of range **NULL** will be returned.
|
2022-12-30 17:29:07 +01:00
|
|
|
|
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
#### SplitElements
|
|
|
|
|
|
|
|
- **uint8_t splitElements(const char \* formula)** split a formula in an internal list of elements.
|
|
|
|
Returns the number of different elements found.
|
2023-04-14 13:34:00 +02:00
|
|
|
Maximum number of elements is hardcoded to 20, which is often enough.
|
2023-04-12 16:52:57 +02:00
|
|
|
- **uint8_t element(uint8_t el)** access the internal list of elements by index el.
|
2023-04-14 13:34:00 +02:00
|
|
|
Note: el should be between 0 and the maximum number returned by **splitElements()**.
|
2023-04-12 16:52:57 +02:00
|
|
|
See example.
|
2022-12-31 17:38:51 +01:00
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
```cpp
|
|
|
|
char formula[24] = "YBa2Cu3O7";
|
|
|
|
nr = splitElements(formula);
|
|
|
|
for (int i = 0; i < nr; i++)
|
|
|
|
{
|
|
|
|
Serial.print(PTOE.massPercentage(formula, element(i)), 3);
|
|
|
|
Serial.print("%\t");
|
|
|
|
Serial.print(element(i));
|
|
|
|
Seriel.println();
|
|
|
|
}
|
|
|
|
```
|
2023-04-12 16:52:57 +02:00
|
|
|
|
|
|
|
#### AtomPercentage
|
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
- **uint32_t count(const char \* formula, const char \* abbreviation = NULL)**
|
|
|
|
- If (abbreviation != NULL) returns the total atoms of one element in a formula.
|
|
|
|
- if (abbreviation == NULL) returns the total atoms of the whole formula.
|
2023-04-12 16:52:57 +02:00
|
|
|
- Returns 0 if it cannot parse the formula given.
|
2023-04-14 13:34:00 +02:00
|
|
|
- **float atomPercentage(const char \* formula, const char \* abbreviation)**
|
2023-04-12 16:52:57 +02:00
|
|
|
Returns atom percentage of the selected element in a formula.
|
|
|
|
|
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
```cpp
|
|
|
|
ap = PTOE.atomPercentage("H2SO4", "O");
|
|
|
|
```
|
|
|
|
|
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
#### Conversion grams moles
|
|
|
|
|
|
|
|
- **float moles2grams(const char \* formula, float moles = 1.0)**
|
|
|
|
Returns the amount of grams needed for a given amount of moles.
|
|
|
|
The default moles == 1, returns the basic conversion factor.
|
|
|
|
- **float grams2moles(const char \* formula, float grams = 1.0)**
|
|
|
|
Returns the amount of moles for a given amount of grams.
|
|
|
|
The default moles == 1, returns the basic conversion factor.
|
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
These functions can be used, e.g. if one wants to solve 2.3 moles of KOH
|
2023-04-14 13:34:00 +02:00
|
|
|
into 10 litres of water to get a defined pH, now much grams I need to weigh?
|
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
```cpp
|
|
|
|
grams = PTOE.moles2grams("KOH", 2.3);
|
|
|
|
```
|
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
#### Weight
|
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
The unit of weight is **Daltons** (Da) or the atomic mass unit (amu), think of it as the number of nucleons.
|
|
|
|
A **Dalton** is defined as 1/12th of the weight of an Carbon atom.
|
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
The **weight(uint8_t element)** call returns the weight of a single atom (by index).
|
2023-04-14 13:34:00 +02:00
|
|
|
The **weight(formula)** call is meant to calculate the weight of a molecule defined by the formula.
|
2023-04-12 16:52:57 +02:00
|
|
|
A molecule is defined as one or more atoms.
|
2022-12-31 18:30:42 +01:00
|
|
|
The functions returns a float, so to get the integer weight, one should use **round()**.
|
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
If the formula can not be parsed, e.g. it contains non existing elements,
|
|
|
|
the **weight()** function will return a weight of 0.
|
2023-04-12 16:52:57 +02:00
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
The **weight(formula, abbreviation)** function is meant to calculate the total weight
|
|
|
|
of one element (by abbreviation) in a molecule.
|
|
|
|
E.g one can weigh the H atoms in H2O (2 of 18).
|
2023-04-12 16:52:57 +02:00
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
```cpp
|
|
|
|
aw = PTOE.weight("H2O", "H");
|
|
|
|
```
|
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
|
|
|
|
#### Formulas
|
2022-12-31 18:30:42 +01:00
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
All element abbreviations are one or two characters long.
|
|
|
|
The first char must be upper case, the (optional) second must be lower case.
|
|
|
|
(except for element 0, n == neutronium, which is added as placeholder).
|
|
|
|
Elements can be followed by a number indicating an amount, no number implies 1.
|
|
|
|
|
|
|
|
Formulas do not care about the order of the atoms.
|
|
|
|
So "C6H6" is equal to "H6C6" or even "CCCCCCHHHHHH" or "C3H3C3H3" etc.
|
|
|
|
|
|
|
|
The formula parsing supports round brackets () to indicate groups in the formula.
|
|
|
|
The library does **not** support square brackets to indicate a group.
|
|
|
|
|
|
|
|
The library does **not** support \*6H20 to indicate hydration.
|
2022-12-31 18:30:42 +01:00
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
Valid formula's might look like:
|
2023-04-14 13:34:00 +02:00
|
|
|
- "B" = single element, Hydrogen, 1 atom.
|
|
|
|
- "Na" = single element, Sodium, 1 atom..
|
|
|
|
- "C6" = single element, multiple times, Benzene.
|
2023-04-15 20:15:18 +02:00
|
|
|
- "H2SO4" = compound molecule, no groups (sulphuric acid).
|
|
|
|
- "C6(COOH)2" = repeating group, (artificial example).
|
|
|
|
- "(H2O)987" = 987 water molecules.
|
|
|
|
- "YBa2Cu3O7" = compound molecule, == some superconductor-ish material.
|
|
|
|
- "Ba((OH)4(COOH)2)3" - recursive repeating groups (artificial example).
|
|
|
|
|
|
|
|
Numbers of an element or a group should be between 1 and 2^32-1 (uint32_t).
|
|
|
|
However in practice values are relative small (< 20).
|
|
|
|
- zero (0) is mapped upon 1.
|
|
|
|
- very large numbers cause overflow when printing the output of some functions.
|
|
|
|
Use - https://github.com/RobTillaart/printHelpers if needed.
|
2022-12-31 17:38:51 +01:00
|
|
|
|
2023-01-02 13:02:36 +01:00
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
#### MassPercentage
|
2023-01-02 13:02:36 +01:00
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
The **massPercentage(formula, abbreviation)** function can determine the percentage of the weight
|
2023-04-12 16:52:57 +02:00
|
|
|
a selected element has in a formula, e.g. the weight of the Oxygen in **H2SO4**.
|
|
|
|
This is calculated by dividing the weight of the element by the total weight.
|
2022-12-31 18:30:42 +01:00
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
```cpp
|
|
|
|
mp = PTOE.massPercentage("H2SO4", "O");
|
|
|
|
```
|
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
If you want to do that for all elements it might be more efficient to calculate the weight
|
|
|
|
of the whole formula once.
|
2022-12-31 18:30:42 +01:00
|
|
|
|
2022-12-31 17:38:51 +01:00
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
#### Avogadro, Dalton, electronVolt.
|
|
|
|
|
|
|
|
The library provides the following (SI) constants:
|
|
|
|
- **const float AVOGADRO = 6.02214076e+23** number of particles in one mole.
|
|
|
|
- **const float DALTON = 1.66053907e-24** weight of one nucleon in grams.
|
|
|
|
- relation: DALTON \* AVOGADRO == 1.0
|
|
|
|
- **const float ELEKTRON_VOLT_JOULE = 1.602176565e-19** eV in Joule
|
|
|
|
- **const float ELEKTRON_VOLT_GRAM = 1.7826619e-39** eV in grams (E=Mc2)
|
|
|
|
- **const float DALTON_JOULE = 1.036427015e5** == DALTON / ELEKTRON_VOLT_JOULE.
|
|
|
|
- **const float DALTON_EV = 931.4940954e12** == DALTON / ELEKTRON_VOLT_GRAM.
|
|
|
|
Can be used to convert the atomic mass to electron volt.
|
|
|
|
|
|
|
|
These constants are not directly used in the library however they fit the scope of the library.
|
|
|
|
There will be functions based upon these constants in the future.
|
|
|
|
|
|
|
|
The **AVOGADRO** constant is the proportionality factor that relates the number of constituent particles
|
|
|
|
(usually molecules, atoms, or ions) in a sample with the amount of substance in that sample.
|
|
|
|
|
|
|
|
From - https://en.wikipedia.org/wiki/Avogadro_constant
|
|
|
|
|
|
|
|
Use https://github.com/RobTillaart/printHelpers to print numbers in the scientific notation.
|
|
|
|
This will prevent printing **OVF** overflow or **0.000**.
|
|
|
|
|
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
#### Debug
|
2022-12-31 17:38:51 +01:00
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
- **float weightFactor()** returns the weightFactor, that was used to
|
|
|
|
minimize the memory used for the elements mass lookup table.
|
2022-12-30 17:29:07 +01:00
|
|
|
|
|
|
|
|
|
|
|
## Future
|
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
#### Must
|
2022-12-30 17:29:07 +01:00
|
|
|
|
|
|
|
- improve documentation
|
2023-04-12 16:52:57 +02:00
|
|
|
- reorganize.
|
|
|
|
|
2022-12-30 17:29:07 +01:00
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
#### Should
|
2022-12-30 17:29:07 +01:00
|
|
|
|
2023-04-15 20:15:18 +02:00
|
|
|
- overload functions
|
|
|
|
- **uint32_t protons(formula)** worker, formula can be single element.
|
|
|
|
- **uint32_t neutrons(formula)** uses protons()
|
|
|
|
- **uint32_t electrons(formula)** uses protons()
|
|
|
|
- add weight of electron as constant. for completeness.
|
|
|
|
- functions around **AVOGADRO**, **DALTON** etc
|
|
|
|
- **float weightEV(formula)**
|
|
|
|
- **float dalton2EV(float Dalton)** to express mass in eV.
|
|
|
|
- **float dalton2Joule(float Dalton)**
|
2023-04-14 13:34:00 +02:00
|
|
|
|
2022-12-31 17:38:51 +01:00
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
#### Could
|
2022-12-31 17:38:51 +01:00
|
|
|
|
2022-12-31 18:30:42 +01:00
|
|
|
- extend unit tests
|
2023-04-15 20:15:18 +02:00
|
|
|
- extend formula parser with error codes.
|
|
|
|
- which ones?
|
|
|
|
- look for optimizations
|
|
|
|
- 3x almost same parser
|
|
|
|
- PROGMEM ?
|
2022-12-30 17:29:07 +01:00
|
|
|
- state table
|
|
|
|
- liquid, gas, solid, unknown (2 bits per element) = ~30 bytes
|
2023-04-12 16:52:57 +02:00
|
|
|
- room temperature + sea level pressure
|
2023-04-15 20:15:18 +02:00
|
|
|
- separate file like elements_name.h
|
2023-04-14 13:34:00 +02:00
|
|
|
- performance **find()**
|
|
|
|
- alphabetical array? tree?
|
2023-04-15 20:15:18 +02:00
|
|
|
- ==> more memory...
|
2023-04-14 13:34:00 +02:00
|
|
|
- support \[] square brackets too.
|
|
|
|
- (NH4)2\[Pt(SCN)6]
|
|
|
|
|
2023-04-12 16:52:57 +02:00
|
|
|
|
|
|
|
#### Wont (unless)
|
|
|
|
|
2023-04-14 13:34:00 +02:00
|
|
|
- case insensitive **find()**
|
|
|
|
element 0 is defined as n conflict with N
|
2023-04-12 16:52:57 +02:00
|
|
|
- support hydrates ?
|
|
|
|
- **Ba(BrO3)2·2H2O** new separator + starts with number.
|
|
|
|
- other liquids than water?
|
|
|
|
- https://en.wikipedia.org/wiki/Glossary_of_chemical_formulae
|
2023-01-02 13:02:36 +01:00
|
|
|
- is there a faster data structure?
|
|
|
|
- search by nr is O(1)
|
|
|
|
- search by name is O(n)
|
2023-04-15 20:15:18 +02:00
|
|
|
- only if more RAM is used. => not
|
2022-12-31 17:38:51 +01:00
|
|
|
- parameters element should be less than \_size
|
|
|
|
- user responsibility
|
2022-12-30 17:29:07 +01:00
|
|
|
- more information?
|
2023-04-14 13:34:00 +02:00
|
|
|
- will not handle isotopes (too much memory needed)
|
2022-12-30 17:29:07 +01:00
|
|
|
- database needed
|
2023-04-14 13:34:00 +02:00
|
|
|
- Electron bands
|
|
|
|
- K L M etc?
|
|
|
|
- valence
|
2022-12-30 17:29:07 +01:00
|
|
|
- temperatures,
|
|
|
|
- melt
|
|
|
|
- evaporate
|
|
|
|
- 2 bytes per temp 4 x 118 = 476 bytes
|
|
|
|
- compression 3 bytes for 2 temps 2x 12 bits = 0..4095 K = 354 bytes
|
|
|
|
|