From ddb27a0216efe60b9bfd6b82841c7fe9bacdfab9 Mon Sep 17 00:00:00 2001 From: Yuval Itzchakov Date: Mon, 10 Sep 2018 13:55:03 +0300 Subject: [PATCH] Add examples to Functor --- core/src/main/scala/cats/Functor.scala | 105 ++++++++++++++++++++----- 1 file changed, 87 insertions(+), 18 deletions(-) diff --git a/core/src/main/scala/cats/Functor.scala b/core/src/main/scala/cats/Functor.scala index d6d0cace523..51422dd7204 100644 --- a/core/src/main/scala/cats/Functor.scala +++ b/core/src/main/scala/cats/Functor.scala @@ -9,7 +9,8 @@ import simulacrum.typeclass * * Must obey the laws defined in cats.laws.FunctorLaws. */ -@typeclass trait Functor[F[_]] extends Invariant[F] { self => +@typeclass trait Functor[F[_]] extends Invariant[F] { + self => def map[A, B](fa: F[A])(f: A => B): F[B] override def imap[A, B](fa: F[A])(f: A => B)(g: B => A): F[B] = map(fa)(f) @@ -17,19 +18,19 @@ import simulacrum.typeclass // derived methods /** - * Alias for [[map]], since [[map]] can't be injected as syntax if - * the implementing type already had a built-in `.map` method. - * - * Example: - * {{{ - * scala> import cats.implicits._ - * - * scala> val m: Map[Int, String] = Map(1 -> "hi", 2 -> "there", 3 -> "you") - * - * scala> m.fmap(_ ++ "!") - * res0: Map[Int,String] = Map(1 -> hi!, 2 -> there!, 3 -> you!) - * }}} - */ + * Alias for [[map]], since [[map]] can't be injected as syntax if + * the implementing type already had a built-in `.map` method. + * + * Example: + * {{{ + * scala> import cats.implicits._ + * + * scala> val m: Map[Int, String] = Map(1 -> "hi", 2 -> "there", 3 -> "you") + * + * scala> m.fmap(_ ++ "!") + * res0: Map[Int,String] = Map(1 -> hi!, 2 -> there!, 3 -> you!) + * }}} + */ final def fmap[A, B](fa: F[A])(f: A => B): F[B] = map(fa)(f) /** @@ -42,38 +43,106 @@ import simulacrum.typeclass * cast is often much more performant. * See [[https://github.com/typelevel/cats/issues/1080#issuecomment-225892635 this example]] * of `widen` creating a `ClassCastException`. + * + * Example: + * {{{ + * scala> import cats.Functor + * scala> import cats.implicits.catsStdInstancesForOption + * + * scala> val s = Some(42) + * scala> Functor[Option].widen(s) + * res0: Option[Int] = Some(42) + * }}} */ def widen[A, B >: A](fa: F[A]): F[B] = fa.asInstanceOf[F[B]] /** * Lift a function f to operate on Functors + * + * Example: + * {{{ + * scala> import cats.Functor + * scala> import cats.implicits.catsStdInstancesForOption + * + * scala> val o = Option(42) + * scala> Functor[Option].lift((x: Int) => x + 10)(o) + * res0: Option[Int] = Some(52) + * }}} */ def lift[A, B](f: A => B): F[A] => F[B] = map(_)(f) /** * Empty the fa of the values, preserving the structure + * + * Example: + * {{{ + * scala> import cats.Functor + * scala> import cats.implicits.catsStdInstancesForList + * + * scala> Functor[List].void(List(1,2,3)) + * res0: List[Unit] = List((), (), ()) + * }}} */ def void[A](fa: F[A]): F[Unit] = as(fa, ()) /** * Tuple the values in fa with the result of applying a function * with the value + * + * Example: + * {{{ + * scala> import cats.Functor + * scala> import cats.implicits.catsStdInstancesForOption + * + * scala> Functor[Option].fproduct(Option(42))(_.toString) + * res0: Option[(Int, String)] = Some((42,42)) + * }}} */ def fproduct[A, B](fa: F[A])(f: A => B): F[(A, B)] = map(fa)(a => a -> f(a)) /** * Replaces the `A` value in `F[A]` with the supplied value. + * + * Example: + * + * {{{ + * scala> import cats.Functor + * scala> import cats.implicits.catsStdInstancesForList + * + * scala> Functor[List].as(List(1,2,3), "hello") + * res0: List[String] = List(hello, hello, hello) + * }}} */ def as[A, B](fa: F[A], b: B): F[B] = map(fa)(_ => b) /** - * Tuples the `A` value in `F[A]` with the supplied `B` value, with the `B` value on the left. - */ + * Tuples the `A` value in `F[A]` with the supplied `B` value, with the `B` value on the left. + * + * Example: + * {{{ + * scala> import scala.collection.immutable.Queue + * scala> import cats.Functor + * scala> import cats.implicits.catsStdInstancesForQueue + * + * scala> Functor[Queue].tupleLeft(Queue("hello", "world"), 42) + * res0: scala.collection.immutable.Queue[(Int, String)] = Queue((42,hello), (42,world)) + * }}} + */ def tupleLeft[A, B](fa: F[A], b: B): F[(B, A)] = map(fa)(a => (b, a)) /** - * Tuples the `A` value in `F[A]` with the supplied `B` value, with the `B` value on the right. - */ + * Tuples the `A` value in `F[A]` with the supplied `B` value, with the `B` value on the right. + * + * Example: + * {{{ + * scala> import scala.collection.immutable.Queue + * scala> import cats.Functor + * scala> import cats.implicits.catsStdInstancesForQueue + * + * scala> Functor[Queue].tupleRight(Queue("hello", "world"), 42) + * res0: scala.collection.immutable.Queue[(String, Int)] = Queue((hello,42), (world,42)) + * }}} + */ def tupleRight[A, B](fa: F[A], b: B): F[(A, B)] = map(fa)(a => (a, b)) def compose[G[_]: Functor]: Functor[λ[α => F[G[α]]]] =