# units [![crates.io](https://img.shields.io/crates/v/units.svg)](https://crates.io/crates/units) [![Build Status](https://travis-ci.org/Boddlnagg/units.svg?branch=master)](https://travis-ci.org/Boddlnagg/units)
**Units of Measure** for Rust. Easy to use, type-safe and customizable.
## Usage (Example)
```rust
#[macro_use] extern crate units;
// It is recommended to put units inside a separate module
pub mod my_units {
// Here we define our unit system with three units
units! {
MyUnits {
Meter[m],
Second[s],
Mile[mile]
}
}
}
// Import the unit constants and also the dimensionless unit `one`
use my_units::f64::{one, m, s, mile};
fn main() {
let km = 1000.*m; // Define the `km` unit as 1000 meter
let h = 60.*60.*s; // Define an hour as 60 * 60 seconds
// These units could also have been defined as separate base units,
// in order to further reduce the risk to mix them up.
// Let's calculate how far a car has moved if it has accelerated
// uniformly from 0 to 100 km/h in 12 seconds
let initial_speed = 0.*km/h;
let final_speed = 100.*km/h;
let time = 12.*s;
let acceleration = (final_speed - initial_speed)/time;
let result = 0.5 * acceleration * time * time; // s = a/2 * t^2
// Here we use debug formatting, which will automatically print the base dimensions
println!("{:?}", result); // This will print `166.66666666666669 m`
// Let's convert the result to miles
let meter_per_mile = 1609.344*m/mile; // This has unit `m/mile`
let result_in_miles = result / meter_per_mile; // This has unit `mile`
// Here we get a dimensionless value by eliminating the unit, then use deref (*) to extract the raw f64.
println!("{} miles", *(result_in_miles/mile)); // This will print `0.103561865372889 miles`
// Now we want to know how long a ball will fall until it reaches the
// floor if dropped from a height of 10 meters
let height = 1.5*m;
let g = 9.81*m/s/s; // Use a gravitational constant of 9.81 m/s^2
let time = (2. * height / g).sqrt(); // sqrt() takes care of the units
// Print the result in milliseconds and round to 2 digits
let ms = 0.001*s; // 1 ms = 0.001 s
println!("The ball falls for {:.2} ms until it hits the ground.", *(time/ms));
// The above will print `The ball falls for 553.00 ms until it hits the ground.`
}
```
Use `cargo run --example basics` to run this example.
<sup>There is an alternative, safer syntax for assigning units to values using function call operator overloading, but this is currently only available in Rust nightly. See [examples/basics.rs](examples/basics.rs) for both versions.</sup>
## How does it work?
The macro invocation shown above will generate a struct `MyUnits<U,T=f64>`, where the first type parameter is a special marker to get the dimension right (it contains exponents for every base unit, encoded as [type-level integers](http://github.com/Boddlnagg/tylar)). `Meter`, `Second` and `Mile` will be type aliases for this marker type with the correct exponents.
The second type parameter denotes the wrapped numeric type (defaults to `f64`). The autogenerated child modules `my_units::f64` and `my_units::f32` then both contain the constants `m`, `s` and `mile`, each of the correct dimension and wrapping the value `1.0`. For example, `f64::m` is of type `MyUnits<Meter, f64>`.
Additionally, the type `One` and the constant `one` are provided for dimensionless values, and only those can be unwrapped using the `Deref` trait (this works automatically for method calls, but sometimes needs to be made explicit using the `*` prefix operator, see the example above).