首页 > 解决方案 > 猫和功能组合物

问题描述

我有数据类型:

import cats.Monoid
import cats.implicits._

object Domain {
  case class Money(amount: Double) extends AnyVal {}

  implicit val moneyMonoid = new Monoid[Money] {
    override def combine(a: Money, b: Money): Money = Money(a.amount + b.amount)
    override def empty: Money                       = Money(0)
  }

  case class Operation(account: Account, amount: Money) extends AnyRef
  type Operations = List[Operation]
}

我想实现totalAmountCalculation类似的功能组合

val calcTotalAmount = map(x => x.amount) >>> combineAll

为此我写了一些代码:

def map[A, B](F: A => B): List[A] => List[B] = (m: List[A]) => m.map(F) 
val combineAll      = moneyMonoid.combineAll _

val calcTotalAmount = map[Operation, Money](x => x.amount) >>> combineAll

我可以使用哪些猫功能来避免编写 Monad 包装器以在函数组合中使用它?

我想看到我的代码:

map(x => x + 1) >>>
filter(x < 100) >>> 
fold(1)((a,b) => a+b))

标签: scalascala-cats

解决方案


如果我正确理解了您的问题,您希望避免使用包装器Amount并且必须x => x.amount在函数定义中的任何地方使用。

有几种方法可以解决这个问题。第一种是使用类型别名,就像您对操作所做的那样。这自然会为您提供Monoid[Int]具有0forzero和添加 for的默认值combine

allMoney
 .map(_ + 1.0)
 .filter(_ > 3.0)
 .foldLeft(1.0)(_ + _)

// 13.0

allMoney.combineAll

// 12.0

另一种选择是使用newtype使用案例类提供类似效果的库:

@newsubtype case class Money(amount: Double)

allMoney
  .map(_ + 1.0)
  .filter(_ > 3.0)
  .foldLeft(1.0)(_ + _)

// 13.0

推荐阅读