Decorum is a Rust library that provides total ordering, equivalence,
hashing, and constraints for floating-point representations. Decorum does not
require std
and can be used in #[no_std]
environments (see the std
feature).
The following total ordering is exposed via traits for primitive types and proxy
types that implement Ord
:
[ -INF < ... < 0 < ... < +INF < NaN ]
IEEE-754 floating-point encoding provides multiple representations of zero (-0
and +0
) and NaN
. This ordering considers all zero and NaN
representations
equal, which differs from the standard partial
ordering.
Decorum exposes several proxy (wrapper) types. Proxy types provide two primary
features: they implement a total ordering with Eq
, Ord
, and Hash
and they
constrain the class of values they can represent. Different type definitions
apply different constraints, with the Total
type applying no constraints at
all.
| Type | Aliases | Numeric Traits | Disallowed Values |
|----------|--------------|--------------------------------------------|-----------------------|
| Total
| | Encoding + Real + Infinite + Nan + Float
| |
| NotNan
| N32
, N64
| Encoding + Real + Infinite
| NaN
|
| Finite
| R32
, R64
| Encoding + Real
| NaN
, -INF
, +INF
|
Proxy types implement common operation traits, such as Add
and Mul
. These
types also implement numeric traits from the
num-traits
crate (such as Float
,
Num
, NumCast
, etc.), in addition to more targeted traits like Real
and
Nan
provided by Decorum.
Constraint violations cause panics. For example, NotNan
is useful for avoiding
or tracing sources of NaN
s in computation, while Total
provides useful
features without introducing any panics at all, because it allows any IEEE-754
floating-point values.
Proxy types should work as a drop-in replacement for primitive types in most
applications, with the most common exception being initialization (because it
requires a conversion). Serialization is optionally supported via
serde (see the serialize-serde
feature).
Numeric traits are essential for generic programming, but the constraints used
by some proxy types prevent them from implementing the ubiquitous Float
trait, because it implies the presence of -INF
, +INF
, and NaN
.
Decorum provides more granular traits that separate these APIs: Real
,
Infinite
, Nan
, and Encoding
. Primitive floating-point types implement all
of these traits and proxy types implement traits that are consistent with their
constraints.
For example, code that wishes to be generic over floating-point types
representing real numbers can use a bound on the Real
trait:
```rust use decorum::Real;
fn f
Both Decorum and num-traits
expose a
Real
trait. Due to some differences in these traits and a poor
interaction with the
num-traits
API, Decorum continues to vendor its own trait. Both traits are
implemented by Decorum where possible.
Proxy types are used via conversions to and from primitive floating-point types and other proxy types.
| Conversion | Input | Output | Violation |
|-----------------|-----------|-----------|-----------|
| from_inner
| primitive | proxy | panic |
| into_inner
| proxy | primitive | n/a |
| from_subset
| proxy | proxy | n/a |
| into_superset
| proxy | proxy | n/a |
The from_inner
and into_inner
conversions move primitive floating-point
values into and out of proxies. The into_superset
and from_subset
conversions provide an inexpensive way to convert between proxy types with
different but compatible constraints. All conversions also support the standard
From
and Into
traits, which can also be applied to literals:
```rust use decorum::R32;
fn f(x: R32) -> R32 { x * 2.0 } let y: R32 = 3.1459.into(); let z = f(2.7182.into()); let w: f32 = z.into(); ```
Proxy types implement Eq
, Hash
, and Ord
, but sometimes it is not
possible or ergonomic to use such a type. Traits can be used with primitive
floating-point values for ordering, equivalence, and hashing instead.
| Trait | Analogous Trait |
|-------------|------------------|
| FloatEq
| Eq
|
| FloatHash
| Hash
|
| FloatOrd
| Ord
|
These traits use the same total ordering and equivalence rules that proxy types do.