Skip to content

Commit

Permalink
Move back to Applicative as replicateA_
Browse files Browse the repository at this point in the history
  • Loading branch information
rabinarai1 committed May 23, 2022
1 parent fc1576b commit c4e960d
Show file tree
Hide file tree
Showing 7 changed files with 50 additions and 49 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -30,5 +30,4 @@ final class MonadOps[F[_], A](private val fa: F[A]) extends AnyVal {
def untilM_(p: F[Boolean])(implicit M: Monad[F]): F[Unit] = M.untilM_(fa)(p)
def iterateWhile(p: A => Boolean)(implicit M: Monad[F]): F[A] = M.iterateWhile(fa)(p)
def iterateUntil(p: A => Boolean)(implicit M: Monad[F]): F[A] = M.iterateUntil(fa)(p)
def replicateM_(n: Int)(implicit M: Monad[F]): F[Unit] = M.replicateM_(n, fa)
}
29 changes: 29 additions & 0 deletions core/src/main/scala/cats/Applicative.scala
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,35 @@ import scala.collection.immutable.IndexedSeq
})(identity)(this))(_.toList)
}

/**
* Given `fa` and `n`, apply `fa` `n` times discarding results to return F[Unit].
*
* Example:
* {{{
* scala> import cats.data.State
*
* scala> type Counter[A] = State[Int, A]
* scala> val getAndIncrement: Counter[Int] = State { i => (i + 1, i) }
* scala> val getAndIncrement5: Counter[Unit] =
* | Applicative[Counter].replicateA_(5, getAndIncrement)
* scala> getAndIncrement5.run(0).value
* res0: (Int, Unit) = (5,())
* }}}
*/
def replicateA_[A](n: Int, fa: F[A]): F[Unit] = {
val fvoid = void(fa)
def loop(n: Int): F[Unit] =
if (n <= 0) unit
else if (n == 1) fvoid
else {
val half = loop(n >> 1)
val both = productR(half)(half)
if ((n & 1) == 1) productR(both)(fvoid)
else both
}
loop(n)
}

/**
* Compose an `Applicative[F]` and an `Applicative[G]` into an
* `Applicative[λ[α => F[G[α]]]]`.
Expand Down
22 changes: 0 additions & 22 deletions core/src/main/scala/cats/Monad.scala
Original file line number Diff line number Diff line change
Expand Up @@ -163,28 +163,6 @@ import simulacrum.{noop, typeclass}

tailRecM(branches.toList)(step)
}

/**
* Given `fa` and `n`, apply `fa` `n` times discarding results to return F[Unit].
*
* Example:
* {{{
* scala> import cats.data.State
*
* scala> type Counter[A] = State[Int, A]
* scala> val getAndIncrement: Counter[Int] = State { i => (i + 1, i) }
* scala> val getAndIncrement5: Counter[Unit] =
* | Monad[Counter].replicateM_(5, getAndIncrement)
* scala> getAndIncrement5.run(0).value
* res0: (Int, Unit) = (5,())
* }}}
*/
def replicateM_[A](n: Int, fa: F[A]): F[Unit] = {
tailRecM(n) { n =>
if (n <= 0) map(unit)(Right.apply)
else map(fa)(_ => Left(n - 1))
}
}
}

object Monad {
Expand Down
1 change: 1 addition & 0 deletions core/src/main/scala/cats/syntax/applicative.scala
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ final class ApplicativeIdOps[A](private val a: A) extends AnyVal {

final class ApplicativeOps[F[_], A](private val fa: F[A]) extends AnyVal {
def replicateA(n: Int)(implicit F: Applicative[F]): F[List[A]] = F.replicateA(n, fa)
def replicateA_(n: Int)(implicit F: Applicative[F]): F[Unit] = F.replicateA_(n, fa)
def unlessA(cond: Boolean)(implicit F: Applicative[F]): F[Unit] = F.unlessA(cond)(fa)
def whenA(cond: Boolean)(implicit F: Applicative[F]): F[Unit] = F.whenA(cond)(fa)
}
16 changes: 15 additions & 1 deletion tests/shared/src/test/scala/cats/tests/ApplicativeSuite.scala
Original file line number Diff line number Diff line change
Expand Up @@ -22,13 +22,15 @@
package cats.tests

import cats.{Align, Applicative, Apply, CoflatMap}
import cats.data.{Const, Validated}
import cats.data.{Const, State, Validated}
import cats.instances.unit
import cats.kernel.Monoid
import cats.kernel.laws.discipline.{MonoidTests, SemigroupTests}
import cats.laws.discipline.arbitrary._
import cats.laws.discipline.{AlignTests, CoflatMapTests}
import cats.syntax.applicative._
import cats.syntax.eq._
import cats.syntax.functor._
import org.scalacheck.Prop._

class ApplicativeSuite extends CatsSuite {
Expand All @@ -39,6 +41,18 @@ class ApplicativeSuite extends CatsSuite {
assert(fa.replicateA(5) === (Some(List(1, 1, 1, 1, 1))))
}

test("replicateA_ executes the Applicative action 'fa' 'n' times") {
val A = Applicative[Option]
val fa = A.pure(0)
assert(fa.replicateA_(5) === Option(unit))

val increment: State[Int, Int] = State { i => (i + 1, i) }

assert(increment.replicateA_(5).runS(0).value === 5)
assert(increment.replicateA_(5).run(0).value === ((5, ())))
assert(increment.replicateA_(5).run(0).value === increment.replicateA(5).void.run(0).value)
}

test("whenA return given argument when cond is true") {
forAll { (l: List[Int]) =>
assert(l.whenA(true) === (List.fill(l.length)(())))
Expand Down
23 changes: 4 additions & 19 deletions tests/shared/src/test/scala/cats/tests/MonadSuite.scala
Original file line number Diff line number Diff line change
Expand Up @@ -21,16 +21,13 @@

package cats.tests

import cats.{Eval, Id, Monad}
import cats.data.{IndexedStateT, State, StateT}
import cats.instances.unit
import cats.data.{IndexedStateT, StateT}
import cats.syntax.apply._
import cats.syntax.monad._
import cats.syntax.applicative._
import cats.syntax.functor._
import org.scalacheck.{Arbitrary, Gen}
import cats.syntax.eq._
import cats.syntax.monad._
import cats.{Eval, Id, Monad}
import org.scalacheck.Prop._
import org.scalacheck.{Arbitrary, Gen}

class MonadSuite extends CatsSuite {
implicit val testInstance: Monad[StateT[Id, Int, *]] = IndexedStateT.catsDataMonadForIndexedStateT[Id, Int]
Expand Down Expand Up @@ -157,16 +154,4 @@ class MonadSuite extends CatsSuite {
assert(actual.value === 2)
}

test("replicateM_ executes the Monad action 'fa' 'n' times") {
val O = Monad[Option]
val fa = O.pure(0)
assert(fa.replicateM_(5) === Option(unit))

val increment: State[Int, Int] = State { i => (i + 1, i) }

assert(increment.replicateM_(5).runS(0).value === 5)
assert(increment.replicateM_(5).run(0).value === ((5,())))
assert(increment.replicateM_(5).run(0).value === increment.replicateA(5).void.run(0).value)
}

}
7 changes: 1 addition & 6 deletions tests/shared/src/test/scala/cats/tests/SyntaxSuite.scala
Original file line number Diff line number Diff line change
Expand Up @@ -411,6 +411,7 @@ object SyntaxSuite {
val a = mock[A]
val fa = a.pure[F]
val replicateA = fa.replicateA(1)
val replicateA_ = fa.replicateA_(1)
}

def testFlatMap[F[_]: FlatMap, A, B, C, D]: Unit = {
Expand Down Expand Up @@ -465,12 +466,6 @@ object SyntaxSuite {
val fea = "meow".raiseError[F, A]
}

def testMonad[F[_],A](implicit F: Monad[F]): Unit = {
val a = mock[A]
val fa = a.pure[F]
fa.replicateM_(5)
}

def testMonadError[F[_, _], E, A, B](implicit F: MonadError[F[E, *], E]): Unit = {
type G[X] = F[E, X]

Expand Down

0 comments on commit c4e960d

Please sign in to comment.