This crate provides traits for comparing and hashing values modulo an equivalence relation specified by a context of user-defined type C
.
The Equivalence
derive macro allows the user to easily implement Equivalence
for custom types.
```rust
/// The equivalence relation mod n over 64-bit unsigned integers struct ModN(u64);
impl PartialEqWith
impl EqWith
impl OrdWith
impl PartialOrdWith
impl HashWith
// Containers can be conveniently compared and hashed modulo a given equivalence context: assert!([1, 2, 3].eqwith(&[4, 5, 6], &ModN(3))); assert!([1, 2, 3].newith(&[4, 5, 6], &ModN(2)));
// The Equivalence
derive macro can be used to derive Equivalence
for custom containers
struct MyPairPartialEqWith
, since they are specified as forwarded
right: u32 // self.right and other.left are compared using PartialEq
, since it is not forwarded
}
assert!(MyPair { left: 5u64, right: 7 }.eqwith(&MyPair { left: 6u64, right: 7 }, &ModN(1))); assert!(MyPair { left: 5u64, right: 7 }.newith(&MyPair { left: 5u64, right: 8 }, &ModN(1)));
// We may also use the macro to derive Equivalence
for a particular context only, with custom logic
struct U32Pair {
#[equiv(fwd = "map_rec |x: &u32, _| x as u64")]
left: u32, // self.left and other.left are first cast to u64, and then compared using PartialEqWith
#[equiv(fwd = "map |x: &u32, ctx: &ModN| (x as u64) % ctx.0")]
right: u32, // right is mapped to right % ctx.0, which is then compared using PartialEq
; this has the same result as the above
}
assert!(U32Pair { left: 3, right: 5 }.eqwith(&U32Pair { left: 5, right: 7 }, &ModN(2))); assert!(U32Pair { left: 3, right: 5 }.newith(&U32Pair { left: 5, right: 7 }, &ModN(3))); ```
The crate is currently quite minimal, but we plan to add:
- Support for no_std
- Better documentation
- Extra forwarding options, such as forwarding to Deref
- Support for comparing collections other than arrays (which are already supported)