Physical units

From HaskellWiki
Revision as of 07:04, 13 September 2016 by AdamGundry (talk | contribs) (link to uom-plugin)
(diff) ← Older revision | Latest revision (diff) | Newer revision → (diff)
Jump to navigation Jump to search

How would one go about modeling units (seconds, meters, meters per second, etc) in Haskell? I'm particularly interested in getting the typechecker to verify proper usage, and do not want to restrict it to any particular numeric representation (i.e. both integral seconds and fractional seconds). If this can in fact be done, it could also be used to model coordinate system axes in, say, Geometric Algebra.

  • It can. Look at Dimensionalized numbers for a toy implementation. I've thought a bit about the Geometric Algebra case, but I can't see a good way of handling it without forcing a choice of basis. I'm also not sure how it would work -- the whole point of GA is to incorporate areas, lengths, volumes, etc. into one number type. I suppose we could use this type of technique to segregate GA spaces with different dimensions or even metrics. --AaronDenney
  • A (non-toy) implementation of statically checked physical dimensions with a complete set of dimensions (but limited number of predefined units) is Dimensional. I apologize for coming up with the same name as Aaron and am on the lookout for a new one. I cannot comment on the applicability to GA. --Bjorn
  • NumericPrelude also contains an implementation of values equipped with physical units. However all unit checking is made dynamically, that is expressions like 1 * meter < 2 * second are accepted by the Haskell compiler but lead to a runtime error. This design allows the programmer to process values from IO world with units, that are unknown at compile time, but it prohibits catching unit errors at compile time. Units are handled as integer vectors of exponents of base units. Computation with these vectors is rather simple, however formatting them is difficult. Formatting and parsing values with units is supported. A small library of SI units is provided.
http://code.haskell.org/numeric-prelude/src/Number/Physical.hs
http://code.haskell.org/numeric-prelude/src/Number/SI.hs
Get with darcs get http://code.haskell.org/numeric-prelude/
  • The quantities package uses an expression parser to create units. For example:
>>> fromString "(1m + 1ft + 2 yard + 50in) * 5 / (8 km/h) => minute"
Right 0.16513500000000006 minute
Of course, this is not verified with the type checker. It uses a custom error type to report issues like undefined units or improper conversions.
  • See also the units package, which uses type families to implement a domain-specific type system for dimension-checked programming. It works only with GHC 7.8+.
  • Measure - Encompasses a few units, currently has build failure on ghc >= 7.0.
  • uom-plugin - uses a type-checker plugin to provide units of measure with good type inference behaviour (requires GHC 7.10+)