Units and Quantities

The astropy.units module gives us the tools to work with units and quantities (values paired with units). The documentation can be found here.

The astropy.units.Unit class is used to represent base units of measurement, such as meters, seconds, kilograms, ect). These units can be manipulated, for example to determine conversion factors from one set of units to another.

You can also define your own units, either as standalone base units or by composing other units together. It is also possible to decompose units into its base units.

As an example, consider the SI unit for mass : kilograms (kg)

from astropy import units as u

u.kg
\[\mathrm{kg}\]

Quantities

Quantities are created by multiplying a float/integer with units, for example a mass:

mass = 3.0*u.kg
mass
\[3 \; \mathrm{kg}\]

The value and unit of the quantity can be accessed separately:

mass.value
3.0
mass.unit
\[\mathrm{kg}\]

Quantities can be combined mathematically, the units combine accordingly. For example, consider the force:

acceleration = 4 * u.m/(u.s*u.s)

force = mass*acceleration
force
\[12 \; \mathrm{\frac{kg\,m}{s^{2}}}\]

Converting Units

Units can be converted to like units. For example, let’s convert the force calculated above into Newtons:

force_N = force.to(u.N)
force_N
\[12 \; \mathrm{N}\]

The conversion above is still within the SI system. We can also convert to units from other systems. For example, converting the force to dynes from the cgs system:

force_dyn = force.to(u.dyn)
force_dyn
\[1200000 \; \mathrm{dyn}\]

Astropy gives us a more convenient way to convert between unit systems without requiring us to do any book keeping. Again, lets convert the force in SI units to the cgs units:

force_cgs = force.cgs
force_cgs
\[1200000 \; \mathrm{dyn}\]

Quantities and NumPy Arrays

Astropy quantities support NumPy arrays. The whole array will have the same units.

import numpy as np

distances = np.array([1, 2, 3, 4]) * u.m
distances
\[[1,~2,~3,~4] \; \mathrm{m}\]
distances.to(u.cm)
\[[100,~200,~300,~400] \; \mathrm{cm}\]
distances/(1 * u.s)
\[[1,~2,~3,~4] \; \mathrm{\frac{m}{s}}\]

Compose Units

Above we converted our force from kg.m.s\(^{-2}\) to N directly. We can let Astropy compose those units for us:

force.unit.compose()
[Unit("N"), Unit("100000 dyn")]

As you can see this gives us all the standard options available.

Decompose Units

You can also decompose units back into base units:

force_N.decompose()
\[12 \; \mathrm{\frac{kg\,m}{s^{2}}}\]

Dimensionless Quantities

Astropy understands dimensionless quantities. For example, lets take the ratio of 2 lengths:

r1 = 10 * u.m
r2 = 2 * u.m

r1 / r2
\[5 \; \mathrm{}\]

Sometimes you will need to decompose units first:

force1 = 3 * u.N
force2 = 6 * u.kg * u.m / u.s / u.s

force_ratio = force1 / force2
force_ratio
\[0.5 \; \mathrm{\frac{s^{2}\,N}{kg\,m}}\]
force_ratio.decompose()
\[0.5 \; \mathrm{}\]

Find Equivalent Units

You can find equivalent units by using the find_equivalent_units() method. For example, for miliseconds (ms):

u.ms.find_equivalent_units()
Primary nameUnit definitionAliases
a3.15576e+07 sannum
d86400 sday
fortnight1.2096e+06 s
h3600 shour, hr
min60 sminute
sirreduciblesecond
sday86164.1 s
wk604800 sweek
yr3.15576e+07 syear

Set Equivalencies

In some contexts we want to use certain units interchangeably, that are not actually equivalent.

For example, the energy of a photon (\(E\)) is related to it’s wavelength (\(\lambda\)) and frequency (\(\nu\)) as:

\[ E = h \nu = h c / \lambda \]

where \(h\) is Planck’s constant. Radio astronomers refer to the energy of an emission line of neutral hydrogen being 1420 MHz or 21 cm. These units are obviously not that of energy, but in this context they can be treated as such.

Astropy allows for this kind of treatment of our units:

wavelength = 21.106 * u.cm

wavelength.to(u.MHz, equivalencies = u.spectral())
\[1420.4134 \; \mathrm{MHz}\]

Finding Equivalent Units with Set Equivalencies

You can also find equivalent within the context of the equivalencies:

freq = 1 * u.Hz

freq.unit.find_equivalent_units(equivalencies = u.spectral())
Primary nameUnit definitionAliases
AU1.49598e+11 mau, astronomical_unit
Angstrom1e-10 mAA, angstrom
Bq1 / sbecquerel
Ci3.7e+10 / scurie
Hz1 / sHertz, hertz
Jkg m2 / s2Joule, joule
Ry2.17987e-18 kg m2 / s2rydberg
cm0.01 mcentimeter
eV1.60218e-19 kg m2 / s2electronvolt
earthRad6.3781e+06 mR_earth, Rearth
erg1e-07 kg m2 / s2
jupiterRad7.1492e+07 mR_jup, Rjup, R_jupiter, Rjupiter
k100 / mKayser, kayser
lyr9.46073e+15 mlightyear
mirreduciblemeter
micron1e-06 m
pc3.08568e+16 mparsec
solRad6.957e+08 mR_sun, Rsun

Fractional Units

Sometimes we’ll need to work with fractional units (units with a fractional power). Floats can work for this, but it is better to use the Standard Library’s fraction.Fraction objects:

from fractions import Fraction

T = 3000.0 * u.K
T
\[3000 \; \mathrm{K}\]
T ** Fraction(3/2)
\[164316.77 \; \mathrm{K^{3/2}}\]

Defining Your Own Units

You can define your own units derived from other units. For example, lets make a unit called a baker’s fortnight (bf), which is a fortnight (14 days) and an extra day:

bakers_fortnight = u.def_unit('bf', 15 * u.day)

bakers_fortnight
\[\mathrm{bf}\]
time = 3 * bakers_fortnight
time
\[3 \; \mathrm{bf}\]
time.to(u.day)
\[45 \; \mathrm{d}\]