Numeric.Units.Dimensional.TF -- Statically checked physical dimensions
Bjorn Buckwalter, bjorn.buckwalter@gmail.com
License: BSD3
= Summary =
In this module we provide data types for performing arithmetic with
physical quantities and units. Information about the physical
dimensions of the quantities/units is embedded in their types and
the validity of operations is verified by the type checker at compile
time. The boxing and unboxing of numerical values as quantities is
done by multiplication and division of units, of which an incomplete
set is provided.
We limit ourselves to "Newtonian" physics. We do not attempt to
accommodate relativistic physics in which e.g. addition of length
and time would be valid.
As far as possible and/or practical the conventions and guidelines
of NIST's "Guide for the Use of the International System of Units
(SI)" [1] are followed. Occasionally we will reference specific
sections from the guide and deviations will be explained.
= Disclaimer =
Merely an engineer, the author doubtlessly uses a language and
notation that makes mathematicians and physicist cringe. He does
not mind constructive criticism (or darcs patches).
The sets of functions and units defined herein are incomplete and
reflect only the author's needs to date. Again, patches are welcome.
The author has elected to keep the module detached from the standard(?)
Haskell library hierarchy. In part because the module name space
layout seems to be an open issue and in part because he is unsure
where to fit it in.
= Preliminaries =
This module requires GHC 7.0 or later. We utilize the following GHC
extensions. Clients of the module are generally not required to use
these extensions.
> {-# LANGUAGE UndecidableInstances
> , ScopedTypeVariables
> , EmptyDataDecls
> , GeneralizedNewtypeDeriving
> , TypeFamilies
> , TypeSynonymInstances
> , FlexibleInstances
> , DeriveDataTypeable
> #-}
> {- |
> Copyright : Copyright (C) 2006-2013 Bjorn Buckwalter
> License : BSD3
>
> Maintainer : bjorn.buckwalter@gmail.com
> Stability : Stable
> Portability: GHC only?
>
> Please refer to the literate Haskell code for documentation of both API
> and implementation.
> -}
> module Numeric.Units.Dimensional.TF
> -- TODO discriminate exports, in particular Variants and Dims.
> where
> import Prelude
> ( Show, Eq, Ord, Enum, Num, Fractional, Floating, RealFloat, Functor, fmap
> , (.), flip, show, (++), undefined, otherwise, (==), String, unwords
> , map, foldr, null, Integer
> )
> import qualified Prelude
> import Data.List (genericLength)
> import Data.Maybe (Maybe (Just, Nothing), catMaybes)
> import Data.Typeable (Typeable)
> import Numeric.NumType.TF
> ( NumType, Zero, toNum, Add, Sub
> , Pos1, Pos2, pos2, Pos3, pos3
> )
> import qualified Numeric.NumType.TF as N (Mul, Div)
We will reuse the operators and function names from the Prelude.
To prevent unpleasant surprises we give operators the same fixity
as the Prelude.
> infixr 8 ^, ^+, ^/, **
> infixl 7 *, /
> infixl 6 +, -
= Dimensional =
Our primary objective is to define a data type that can be used to
represent (while still differentiating between) units and quantities.
There are two reasons for consolidating units and quantities in one
data type. The first being to allow code reuse as they are largely
subject to the same operations. The second being that it allows
reuse of operators (and functions) between the two without resorting
to occasionally cumbersome type classes.
We call this data type 'Dimensional' to capture the notion that the
units and quantities it represents have physical dimensions.
> newtype Dimensional v d a = Dimensional a deriving (Eq, Ord, Enum, Typeable)
The type variable 'a' is the only non-phantom type variable and
represents the numerical value of a quantity or the scale (w.r.t.
SI units) of a unit. For SI units the scale will always be 1. For
non-SI units the scale is the ratio of the unit to the SI unit with
the same physical dimension.
Since 'a' is the only non-phantom type we were able to define
'Dimensional' as a newtype, avoiding boxing at runtime.
= The variety 'v' of 'Dimensional' =
The phantom type variable v is used to distinguish between units
and quantities. It should be one of the following:
> data DUnit deriving Typeable
> data DQuantity deriving Typeable
For convenience we define type synonyms for units and quantities.
> type Unit = Dimensional DUnit
> type Quantity = Dimensional DQuantity
The relationship between (the value of) a 'Quantity', its numerical
value and its 'Unit' is described in 7.1 "Value and numerical value
of a quantity" of [1]. In short a 'Quantity' is the product of a
number and a 'Unit'. We define the '(*~)' operator as a convenient
way to declare quantities as such a product.
> (*~) :: Num a => a -> Unit d a -> Quantity d a
> x *~ Dimensional y = Dimensional (x Prelude.* y)
Conversely, the numerical value of a 'Quantity' is obtained by
dividing the 'Quantity' by its 'Unit' (any unit with the same
physical dimension). The '(/~)' operator provides a convenient way
of obtaining the numerical value of a quantity.
> (/~) :: Fractional a => Quantity d a -> Unit d a -> a
> Dimensional x /~ Dimensional y = x Prelude./ y
We give '*~' and '/~' the same fixity as '*' and '/' defined below.
Note that this necessitates the use of parenthesis when composing
units using '*' and '/', e.g. "1 *~ (meter / second)".
> infixl 7 *~, /~
= The dimension 'd' of 'Dimensional' =
The phantom type variable d encompasses the physical dimension of
the 'Dimensional'. As detailed in [5] there are seven base dimensions,
which can be combined in integer powers to a given physical dimension.
We represent physical dimensions as the powers of the seven base
dimensions that make up the given dimension. The powers are represented
using NumTypes. For convenience we collect all seven base dimensions
in a data type 'Dim'.
> data Dim l m t i th n j deriving Typeable
where the respective dimensions are represented by type variables
using the following convention.
l -- Length
m -- Mass
t -- Time
i -- Electric current
th -- Thermodynamic temperature
n -- Amount of substance
j -- Luminous intensity
We could have chosen to provide type variables for the seven base
dimensions in 'Dimensional' instead of creating a new data type
'Dim'. However, that would have made any type signatures involving
'Dimensional' very cumbersome. By encompassing the physical dimension
in a single type variable we can "hide" the cumbersome type arithmetic
behind convenient type families as will be seen later.
Using our 'Dim' data type we define some type synonyms for convenience
and illustrative purposes. We start with the base dimensions.
> type DOne = Dim Zero Zero Zero Zero Zero Zero Zero
> type DLength = Dim Pos1 Zero Zero Zero Zero Zero Zero
> type DMass = Dim Zero Pos1 Zero Zero Zero Zero Zero
> type DTime = Dim Zero Zero Pos1 Zero Zero Zero Zero
> type DElectricCurrent = Dim Zero Zero Zero Pos1 Zero Zero Zero
> type DThermodynamicTemperature = Dim Zero Zero Zero Zero Pos1 Zero Zero
> type DAmountOfSubstance = Dim Zero Zero Zero Zero Zero Pos1 Zero
> type DLuminousIntensity = Dim Zero Zero Zero Zero Zero Zero Pos1
Using the above type synonyms we can define type synonyms for
quantities of particular physical dimensions.
Quantities with the base dimensions.
> type Dimensionless = Quantity DOne
> type Length = Quantity DLength
> type Mass = Quantity DMass
> type Time = Quantity DTime
> type ElectricCurrent = Quantity DElectricCurrent
> type ThermodynamicTemperature = Quantity DThermodynamicTemperature
> type AmountOfSubstance = Quantity DAmountOfSubstance
> type LuminousIntensity = Quantity DLuminousIntensity
= Arithmetic on physical dimensions =
When performing arithmetic on units and quantities the arithmetics
must be applied to both the numerical values of the Dimensionals
but also to their physical dimensions. The type level arithmetic
on physical dimensions is governed by type families.
Multiplication of dimensions corresponds to adding of the base
dimensions' exponents.
> type family Mul a b
> type instance Mul (Dim l m t i th n j)
> (Dim l' m' t' i' th' n' j')
> = Dim (Add l l')
> (Add m m')
> (Add t t')
> (Add i i')
> (Add th th')
> (Add n n')
> (Add j j')
Division of dimensions corresponds to subtraction of the base
dimensions' exponents.
> type family Div a b
> type instance Div (Dim l m t i th n j)
> (Dim l' m' t' i' th' n' j')
> = Dim (Sub l l')
> (Sub m m')
> (Sub t t')
> (Sub i i')
> (Sub th th')
> (Sub n n')
> (Sub j j')
We limit ourselves to integer powers of Dimensionals as fractional
powers make little physical sense. Since the value of the exponent
affects the type of the result the value of the exponent must be
visible to the type system, therefore we will generally represent
the exponent with a 'NumType'.
Powers of dimensions corresponds to multiplication of the base
dimensions' exponents by the exponent.
> type family Pow d x
> type instance Pow (Dim l m t i th n j) x
> = Dim (N.Mul l x)
> (N.Mul m x)
> (N.Mul t x)
> (N.Mul i x)
> (N.Mul th x)
> (N.Mul n x)
> (N.Mul j x)
Roots of dimensions corresponds to division of the base dimensions'
exponents by order(?) of the root.
> type family Root d x
> type instance Root (Dim l m t i th n j) x
> = Dim (N.Div l x)
> (N.Div m x)
> (N.Div t x)
> (N.Div i x)
> (N.Div th x)
> (N.Div n x)
> (N.Div j x)
= Arithmetic on units and quantities =
Thanks to the arithmetic on physical dimensions having been sorted
out separately a lot of the arithmetic on Dimensionals is straight
forward. In particular the type signatures are much simplified.
Multiplication, division and powers apply to both units and quantities.
> (*) :: (Num a) => Dimensional v d a -> Dimensional v d' a
> -> Dimensional v (Mul d d') a
> Dimensional x * Dimensional y = Dimensional (x Prelude.* y)
> (/) :: (Fractional a) => Dimensional v d a -> Dimensional v d' a
> -> Dimensional v (Div d d') a
> Dimensional x / Dimensional y = Dimensional (x Prelude./ y)
> (^) :: (Fractional a, NumType n) => Dimensional v d a -> n
> -> Dimensional v (Pow d n) a
> Dimensional x ^ n = Dimensional (x Prelude.^^ (toNum n :: Integer))
In the unlikely case someone needs to use this library with
non-fractional numbers we provide the alternative power operator
'^+' that is restricted to positive exponents.
> (^+) :: (Num a, NumType n) => Dimensional v d a -> n
> -> Dimensional v (Pow d n) a
> Dimensional x ^+ n = Dimensional (x Prelude.^ (toNum n :: Integer))
A special case is that dimensionless quantities are not restricted
to integer exponents. This is accommodated by the '**' operator
defined later.
= Quantity operations =
Some additional operations obviously only make sense for quantities.
Of these, negation, addition and subtraction are particularly simple
as they are done in a single physical dimension.
> negate :: (Num a) => Quantity d a -> Quantity d a
> negate (Dimensional x) = Dimensional (Prelude.negate x)
> (+) :: (Num a) => Quantity d a -> Quantity d a -> Quantity d a
> Dimensional x + Dimensional y = Dimensional (x Prelude.+ y)
> (-) :: (Num a) => Quantity d a -> Quantity d a -> Quantity d a
> x - y = x + negate y
Absolute value.
> abs :: (Num a) => Quantity d a -> Quantity d a
> abs (Dimensional x) = Dimensional (Prelude.abs x)
Roots of arbitrary (integral) degree. Appears to occasionally be useful
for units as well as quantities.
> nroot :: (Floating a, NumType n) => n -> Dimensional v d a
> -> Dimensional v (Root d n) a
> nroot n (Dimensional x) = Dimensional (x Prelude.** (1 Prelude./ toNum n))
We provide short-hands for the square and cubic roots.
> sqrt :: (Floating a) => Dimensional v d a -> Dimensional v (Root d Pos2) a
> sqrt = nroot pos2
> cbrt :: (Floating a) => Dimensional v d a -> Dimensional v (Root d Pos3) a
> cbrt = nroot pos3
We also provide an operator alternative to nroot for those that
prefer such.
> (^/) :: (Floating a, NumType n) => Dimensional v d a -> n
> -> Dimensional v (Root d n) a
> (^/) = flip nroot
= List functions =
Here we define operators and functions to make working with homogenuous
lists of dimensionals more convenient.
We define two convenience operators for applying units to all
elements of a functor (e.g. a list).
> (*~~) :: (Functor f, Num a) => f a -> Unit d a -> f (Quantity d a)
> xs *~~ u = fmap (*~ u) xs
> (/~~) :: (Functor f, Fractional a) => f (Quantity d a) -> Unit d a -> f a
> xs /~~ u = fmap (/~ u) xs
> infixl 7 *~~, /~~
The sum of all elements in a list.
> sum :: forall d a . Num a => [Quantity d a] -> Quantity d a
> sum = foldr (+) _0
The length of the list as a 'Dimensionless'. This can be useful for
purposes of e.g. calculating averages.
> dimensionlessLength :: Num a => [Dimensional v d a] -> Dimensionless a
> dimensionlessLength = Dimensional . genericLength
= Dimensionless =
For dimensionless quantities pretty much any operation is applicable.
We provide this freedom by making 'Dimensionless' an instance of
'Functor'.
> instance Functor Dimensionless where
> fmap f (Dimensional x) = Dimensional (f x)
We continue by defining elementary functions on 'Dimensionless'
that may be obviously useful.
> exp, log, sin, cos, tan, asin, acos, atan, sinh, cosh, tanh, asinh, acosh, atanh
> :: (Floating a) => Dimensionless a -> Dimensionless a
> exp = fmap Prelude.exp
> log = fmap Prelude.log
> sin = fmap Prelude.sin
> cos = fmap Prelude.cos
> tan = fmap Prelude.tan
> asin = fmap Prelude.asin
> acos = fmap Prelude.acos
> atan = fmap Prelude.atan
> sinh = fmap Prelude.sinh
> cosh = fmap Prelude.cosh
> tanh = fmap Prelude.tanh
> asinh = fmap Prelude.asinh
> acosh = fmap Prelude.acosh
> atanh = fmap Prelude.atanh
> (**) :: (Floating a)
> => Dimensionless a -> Dimensionless a -> Dimensionless a
> Dimensional x ** Dimensional y = Dimensional (x Prelude.** y)
For 'atan2' the operands need not be dimensionless but they must be
of the same type. The result will of course always be dimensionless.
> atan2 :: (RealFloat a)
> => Quantity d a -> Quantity d a -> Dimensionless a
> atan2 (Dimensional y) (Dimensional x) = Dimensional (Prelude.atan2 y x)
The only unit we will define in this module is 'one'. The unit one
has dimension one and is the base unit of dimensionless values. As
detailed in 7.10 "Values of quantities expressed simply as numbers:
the unit one, symbol 1" of [1] the unit one generally does not
appear in expressions. However, for us it is necessary to use 'one'
as we would any other unit to perform the "boxing" of dimensionless
values.
> one :: Num a => Unit DOne a
> one = Dimensional 1
For convenience we define some constants for small integer values
that often show up in formulae. We also throw in 'pi' and 'tau' for
good measure.
The constant for zero is polymorphic as proposed by Douglas McClean
(http://code.google.com/p/dimensional/issues/detail?id=39) allowing
it to express zero Length or Capacitance or Velocity etc, in addition
to the dimensionless value zero.
> _0 :: (Num a) => Quantity d a
> _0 = Dimensional 0
> _1, _2, _3, _4, _5, _6, _7, _8, _9 :: (Num a) => Dimensionless a
> _1 = 1 *~ one
> _2 = 2 *~ one
> _3 = 3 *~ one
> _4 = 4 *~ one
> _5 = 5 *~ one
> _6 = 6 *~ one
> _7 = 7 *~ one
> _8 = 8 *~ one
> _9 = 9 *~ one
For background on 'tau' see http://tauday.com/tau-manifesto (but also
feel free to review http://www.thepimanifesto.com).
> pi, tau :: (Floating a) => Dimensionless a
> pi = Prelude.pi *~ one
> tau = _2 * pi
= Instances of 'Show' =
We will conclude by providing a reasonable 'Show' instance for
quantities. We neglect units since it is unclear how to represent them
in a way that distinguishes them from quantities, or whether that is
even a requirement.
> instance forall d a. (Show d, Show a) => Show (Quantity d a) where
> show (Dimensional x) = show x ++ if (null unit) then "" else " " ++ unit
> where unit = show (undefined :: d)
The above implementation of 'show' relies on the dimension 'd' being an
instance of 'Show'. The "normalized" unit of the quantity can be inferred
from its dimension.
> instance forall l m t i th n j.
> ( NumType l
> , NumType m
> , NumType t
> , NumType i
> , NumType th
> , NumType n
> , NumType j
> ) => Show (Dim l m t i th n j) where
> show _ = (unwords . catMaybes)
> [ dimUnit "m" (undefined :: l)
> , dimUnit "kg" (undefined :: m)
> , dimUnit "s" (undefined :: t)
> , dimUnit "A" (undefined :: i)
> , dimUnit "K" (undefined :: th)
> , dimUnit "mol" (undefined :: n)
> , dimUnit "cd" (undefined :: j)
> ]
The helper function 'dimUnit' defined next conditions a 'String' (unit)
with an exponent, if appropriate. The reason we define 'dimUnit' at the
top-level rather than in the where-clause is that it may be useful for
users of the 'Extensible' module.
> dimUnit :: (NumType n) => String -> n -> Maybe String
> dimUnit u n
> | x == 0 = Nothing
> | x == 1 = Just u
> | otherwise = Just (u ++ "^" ++ show x)
> where x = toNum n :: Integer
= The 'prefix' function =
We will define a 'prefix' function which applies a scale factor to
a unit. The 'prefix' function will be used by other modules to
define the SI prefixes and non-SI units.
> prefix :: (Num a) => a -> Unit d a -> Unit d a
> prefix x (Dimensional y) = Dimensional (x Prelude.* y)
= Conclusion and usage =
We have defined operators and units that allow us to define and
work with physical quantities. A physical quantity is defined by
multiplying a number with a unit (the type signature is optional).
] v :: Velocity Prelude.Double
] v = 90 *~ (kilo meter / hour)
It follows naturally that the numerical value of a quantity is
obtained by division by a unit.
] numval :: Prelude.Double
] numval = v /~ (meter / second)
The notion of a quantity as the product of a numerical value and a
unit is supported by 7.1 "Value and numerical value of a quantity" of
[1]. While the above syntax is fairly natural it is unfortunate that
it must violate a number of the guidelines in [1], in particular 9.3
"Spelling unit names with prefixes", 9.4 "Spelling unit names obtained
by multiplication", 9.5 "Spelling unit names obtained by division".
As a more elaborate example of how to use the module we define a
function for calculating the escape velocity of a celestial body
[2].
] escapeVelocity :: (Floating a) => Mass a -> Length a -> Velocity a
] escapeVelocity m r = sqrt (two * g * m / r)
] where
] two = 2 *~ one
] g = 6.6720e-11 *~ (newton * meter ^ pos2 / kilo gram ^ pos2)
The following is an example GHC session where the above function
is used to calculate the escape velocity of Earth in kilometer per
second.
*Numeric.Dimensional> :set +t
*Numeric.Dimensional> let me = 5.9742e24 *~ kilo gram -- Mass of Earth.
me :: Quantity DMass GHC.Float.Double
*Numeric.Dimensional> let re = 6372.792 *~ kilo meter -- Mean radius of Earth.
re :: Quantity DLength GHC.Float.Double
*Numeric.Dimensional> let ve = escapeVelocity me re -- Escape velocity of Earth.
ve :: Velocity GHC.Float.Double
*Numeric.Dimensional> ve /~ (kilo meter / second)
11.184537332296259
it :: GHC.Float.Double
For completeness we should also show an example of the error messages
we will get from GHC when performing invalid arithmetic. In the
best case GHC will be able to use the type synonyms we have defined
in its error messages. In other cases the error messages aren't very
friendly.
] x = 1 *~ meter + 1 *~ second
Couldn't match expected type `Numeric.NumType.TF.S
Numeric.NumType.TF.Z'
with actual type `Numeric.NumType.TF.Z'
Expected type: Unit DLength a1
Actual type: Unit DTime a0
In the second argument of `(*~)', namely `second'
In the second argument of `(+)', namely `1 *~ second'
] x = 1 *~ meter / (1 *~ second) + 1 *~ kilo gram
1 *~ meter / (1 *~ second) + 1 *~ kilo gram
Couldn't match type `Numeric.NumType.TF.S Numeric.NumType.TF.Z'
with `Numeric.NumType.TF.Zero'
Expected type: Quantity DMass Double
Actual type: Numeric.Units.Dimensional.TF.Dimensional
DQuantity (Div DLength DTime) Double
In the first argument of `(+)', namely `1 *~ meter / (1 *~ second)'
In the expression: 1 *~ meter / (1 *~ second) + 1 *~ kilo gram
Couldn't match type `Numeric.NumType.TF.Z'
with `Numeric.NumType.TF.Pos1'
Expected type: Quantity DMass Double
Actual type: Numeric.Units.Dimensional.TF.Dimensional
DQuantity (Div DLength DTime) Double
In the first argument of `(+)', namely `1 *~ meter / (1 *~ second)'
In the expression: 1 *~ meter / (1 *~ second) + 1 *~ kilo gram
Couldn't match type `Numeric.NumType.TF.N
(Numeric.NumType.TF.S Numeric.NumType.TF.Z)'
with `Numeric.NumType.TF.Zero'
Expected type: Quantity DMass Double
Actual type: Numeric.Units.Dimensional.TF.Dimensional
DQuantity (Div DLength DTime) Double
In the first argument of `(+)', namely `1 *~ meter / (1 *~ second)'
In the expression: 1 *~ meter / (1 *~ second) + 1 *~ kilo gram
It is the author's experience that the usefullness of the compiler
error messages is more often than not limited to pinpointing the
location of errors.
= Related work =
This module is a port of the Numeric.Units.Dimensional module in
the dimensional [3] package. This module differs from
Numeric.Units.Dimensional in that the dimension tracking is implemented
using type families rather than functional dependencies.
Henning Thielemann numeric prelude has a physical units library,
however, checking of dimensions is dynamic rather than static.
Aaron Denney has created a toy example of statically checked
physical dimensions covering only length and time. HaskellWiki
has pointers [4] to these.
Also see Samuel Hoffstaetter's blog post [5] which uses techniques
similar to this library.
Libraries with similar functionality exist for other programming
languages and may serve as inspiration. The author has found the
Java library JScience [6] and the Fortress programming language [7]
particularly noteworthy.
= References =
[1] http://physics.nist.gov/Pubs/SP811/
[2] http://en.wikipedia.org/wiki/Escape_velocity
[3] http://dimensional.googlecode.com
[4] http://www.haskell.org/haskellwiki/Physical_units
[5] http://liftm.wordpress.com/2007/06/03/scientificdimension-type-arithmetic-and-physical-units-in-haskell/
[6] http://jscience.org/
[7] http://research.sun.com/projects/plrg/fortress.pdf