mirror of
https://github.com/mpusz/mp-units.git
synced 2025-08-03 04:14:27 +02:00
docs: "System of Units" chapter added
This commit is contained in:
@@ -0,0 +1,190 @@
|
||||
# Systems of Units
|
||||
|
||||
Modeling a [system of units](../../../appendix/glossary/#system-of-units) is probably
|
||||
the most important feature and a selling point of every physical units library.
|
||||
Thanks to that, the library can protect users from performing invalid operations on
|
||||
quantities and provide automated conversion factors between various compatible units.
|
||||
|
||||
Probably all the libraries in the wild model the [SI](../../../appendix/glossary/#si)
|
||||
and many of them provide support for additional units belonging to various other systems
|
||||
(i.e. imperial).
|
||||
|
||||
|
||||
## Systems of Units are based on Systems of Quantities
|
||||
|
||||
[Systems of quantities](../../../appendix/glossary/#system-of-quantities) specify a set
|
||||
of quantities and equations relating to those quantities. Those equations do not take any
|
||||
unit or a numerical representation into account at all. In order to create a quantity,
|
||||
we need to add those missing pieces of information. This is where
|
||||
a [system of units](../../../appendix/glossary/#system-of-units) kicks in.
|
||||
|
||||
The [SI](../../../appendix/glossary/#si) is explicitly stated to be based on
|
||||
the [ISQ](../../../appendix/glossary/#isq). Among others, it defines
|
||||
`7` [base units](../../../appendix/glossary/#base-unit), one for each
|
||||
[base quantity](../../../appendix/glossary/#base-quantity). In the **mp-units**
|
||||
this is expressed by associating a quantity kind (that we discussed in detail in the
|
||||
previous chapter) with a unit that is used to express it:
|
||||
|
||||
```cpp
|
||||
inline constexpr struct metre : named_unit<"m", kind_of<isq::length>> {} metre;
|
||||
```
|
||||
|
||||
!!! note
|
||||
|
||||
The `kind_of<isq::length>` above states explicitly that this unit has
|
||||
an associated quantity kind. In other words, `si::metre` (and scaled units based
|
||||
on it) can be used to express the amount of any quantity of kind length.
|
||||
|
||||
|
||||
## Units compose
|
||||
|
||||
One of the strongest points of the [SI](../../../appendix/glossary/#si) system
|
||||
is that its units compose. This allows providing thousands of different units for
|
||||
hundreds of various quantities with a really small set of predefined units
|
||||
and prefixes.
|
||||
|
||||
The same is modeled in the **mp-units** library, which also allows composing
|
||||
predefined units to create a nearly infinite number of different
|
||||
[derived units](../../../appendix/glossary/#derived-unit). For example, one can write:
|
||||
|
||||
```cpp
|
||||
quantity<si::metre / si::second> q;
|
||||
```
|
||||
|
||||
to express a quantity of speed. The resulting quantity type is implicitly inferred
|
||||
from the [unit equation](../../../appendix/glossary/#unit-equation) by repeating
|
||||
exactly the same operations on the associated quantity kinds.
|
||||
|
||||
|
||||
## Many shades of the same unit
|
||||
|
||||
The [SI](../../../appendix/glossary/#si) provides the names for 22 common
|
||||
[coherent units](../../../appendix/glossary/#coherent-derived-unit) of 22
|
||||
[derived quantities](../../../appendix/glossary/#derived-quantity).
|
||||
|
||||
Each such named [derived unit](../../../appendix/glossary/#derived-unit) is a result
|
||||
of a specific predefined [unit equation](../../../appendix/glossary/#unit-equation).
|
||||
For example, a unit of power quantity is defined in the library as:
|
||||
|
||||
```cpp
|
||||
inline constexpr struct watt : named_unit<"W", joule / second> {} watt;
|
||||
```
|
||||
|
||||
However, a power quantity can be expressed in other units as well. For example,
|
||||
the following:
|
||||
|
||||
```cpp
|
||||
auto q1 = 42 * W;
|
||||
std::cout << q1 << "\n";
|
||||
std::cout << q1[J / s] << "\n";
|
||||
std::cout << q1[N * m / s] << "\n";
|
||||
std::cout << q1[kg * m2 / s3] << "\n";
|
||||
```
|
||||
|
||||
prints:
|
||||
|
||||
```text
|
||||
42 W
|
||||
42 J/s
|
||||
42 N m/s
|
||||
42 kg m²/s³
|
||||
```
|
||||
|
||||
All of the above quantities are equivalent and mean exactly the same.
|
||||
|
||||
|
||||
## Constraining a derived unit to work only with a specific derived quantity
|
||||
|
||||
Some derived units are valid only for specific derived quantities. For example,
|
||||
SI specifies both `hertz` and `becquerel` derived units with the same unit equation `1 / s`.
|
||||
However, it also explicitly states:
|
||||
|
||||
!!! quote "SI Brochure"
|
||||
|
||||
The hertz shall only be used for periodic phenomena and the becquerel shall only be used for
|
||||
stochastic processes in activity referred to a radionuclide.
|
||||
|
||||
The library allows constraining such units in the following way:
|
||||
|
||||
```cpp
|
||||
inline constexpr struct hertz : named_unit<"Hz", 1 / second, kind_of<isq::frequency>> {} hertz;
|
||||
inline constexpr struct becquerel : named_unit<"Bq", 1 / second, kind_of<isq::activity>> {} becquerel;
|
||||
```
|
||||
|
||||
With the above, `hertz` can only be used for frequencies while becquerel should only be used for
|
||||
quantities of activity. This means that the following equation will not compile:
|
||||
|
||||
```cpp
|
||||
auto q = 1 * Hz + 1 * Bq; // Fails to compile
|
||||
```
|
||||
|
||||
This is exactly what we wanted to achieve to improve the type-safety of the library.
|
||||
|
||||
|
||||
## Prefixed units
|
||||
|
||||
Besides named units, the [SI](../../../appendix/glossary/#si) specifies also 24 prefixes
|
||||
(all being a power of `10`) that can be prepended to all named units to obtain various scaled
|
||||
versions of them.
|
||||
|
||||
Implementation of `std::ratio` provided by all major compilers is able to express only
|
||||
16 of them. This is why, in the **mp-units**, we had to find an alternative way to represent
|
||||
unit magnitude in a more flexible way.
|
||||
|
||||
Each prefix is implemented as:
|
||||
|
||||
```cpp
|
||||
template<PrefixableUnit auto U> struct quecto_ : prefixed_unit<"q", mag_power<10, -30>, U> {};
|
||||
template<PrefixableUnit auto U> inline constexpr quecto_<U> quecto;
|
||||
```
|
||||
|
||||
and then a [PrefixableUnit](../basic_concepts/#prefixableunit) can be prefixed in the following
|
||||
way:
|
||||
|
||||
```cpp
|
||||
inline constexpr auto qm = quecto<metre>;
|
||||
```
|
||||
|
||||
The usage of `mag_power` not only enables providing support for SI prefixes but it can also
|
||||
efficiently represent any rational magnitude. For example, IEC 80000 prefixes used in the
|
||||
IT industry can be implemented as:
|
||||
|
||||
```cpp
|
||||
template<PrefixableUnit auto U> struct yobi_ : prefixed_unit<"Yi", mag_power<2, 80>, U> {};
|
||||
template<PrefixableUnit auto U> inline constexpr yobi_<U> yobi;
|
||||
```
|
||||
|
||||
## Scaled units
|
||||
|
||||
In the [SI](../../../appendix/glossary/#si), all units are either base or derived units or prefixed
|
||||
versions of those. However, those are not the only options possible.
|
||||
|
||||
For example, there is a list of [off-system units](../../../appendix/glossary/#off-system-unit)
|
||||
accepted for use with SI. All of those are scaled versions of the SI units with ratios that can't
|
||||
be explicitly expressed with predefined SI prefixes. Those include units like minute, hour, or
|
||||
electronvolt:
|
||||
|
||||
```cpp
|
||||
inline constexpr struct minute : named_unit<"min", mag<60> * si::second> {} minute;
|
||||
inline constexpr struct hour : named_unit<"h", mag<60> * minute> {} hour;
|
||||
inline constexpr struct electronvolt : named_unit<"eV", mag<ratio{1'602'176'634, 1'000'000'000}> * mag_power<10, -19> * si::joule> {} electronvolt;
|
||||
```
|
||||
|
||||
Also, units of other [systems of units](../../../appendix/glossary/#system-of-units) are often defined
|
||||
in terms of scaled versions of the SI units. For example, the international yard is defined as:
|
||||
|
||||
```cpp
|
||||
inline constexpr struct yard : named_unit<"yd", mag<ratio{9'144, 10'000}> * si::metre> {} yard;
|
||||
```
|
||||
|
||||
For some units, a magnitude might also be irrational. The best example here is a `degree` which
|
||||
is defined using a floating-point magnitude having a factor of the number π (Pi):
|
||||
|
||||
```cpp
|
||||
inline constexpr struct mag_pi : magnitude<std::numbers::pi_v<long double>> {} mag_pi;
|
||||
```
|
||||
|
||||
```cpp
|
||||
inline constexpr struct degree : named_unit<basic_symbol_text{"°", "deg"}, mag_pi / mag<180> * si::radian> {} degree;
|
||||
```
|
||||
|
||||
|
Reference in New Issue
Block a user