This crate provides functors in Rust, including
implementations for all collections in
std::collections
.
Functor
impl<'a, A, B> Functor<'a, B> for Option<A>
where
A: 'a,
B: 'a,
{
type Inner = A;
type Mapped<'b> = Option<B>
where
'a: 'b,
B: 'b;
type Map<'b, C> = Option<C>
where
'a: 'b,
C: 'a;
fn fmap<'b, F>(self, f: F) -> Option<B>
where
'a: 'b,
F: Fn(A) -> B + 'b,
{
self.map(f)
}
}
Functor::fmap
``` use fmap::Functor;
let ok: Result
let err: Result
Note that the compiler can't infer that mapping an inner type to itself doesn't change the wrapped type:
//fn double_inner_i32<'a, T>(x: T) -> T // doesn't work
fn double_inner_i32<'a, T>(x: T) -> T::Mapped<'a>
where
T: Functor<'a, i32, Inner = i32>,
{
x.fmap(|x| 2 * x)
}
This can either be fixed with an extra bound:
fn double_inner_i32<'a, T>(x: T) -> T
where
//T: Functor<'a, i32, Inner = i32>, // doesn't work
T: Functor<'a, i32, Inner = i32, Mapped<'a> = T>,
{
x.fmap(|x| 2 * x)
}
Or it can be fixed using the zero-cost Functor::from_mapped
helper function:
fn double_inner_i32<'a, T>(x: T) -> T
where
T: Functor<'a, i32, Inner = i32>,
{
//x.fmap(|x| 2 * x) // doesn't work
T::from_mapped(x.fmap(|x| 2 * x))
}