scala - 猫和功能组合物
问题描述
我有数据类型:
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))
解决方案
如果我正确理解了您的问题,您希望避免使用包装器Amount
并且必须x => x.amount
在函数定义中的任何地方使用。
有几种方法可以解决这个问题。第一种是使用类型别名,就像您对操作所做的那样。这自然会为您提供Monoid[Int]
具有0
forzero
和添加 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
推荐阅读
- android - 为什么 ExoPlayer 的视频初始加载时间超过 5 秒?
- qlikview - 如何循环过去 4 周的数据
- node.js - 如何在nodejs中读取大型XLSX文件
- javascript - Vue JS - 我如何滚动(聚焦)到“活动”li 元素?
- dart - Flutter:自定义 Stepper 小部件需要 Material 小部件祖先
- ajax - 如何使用 Ajax 访问数组单选列表元素的值
- css - 如何将 vue 数组的各个元素绑定到一个类
- c# - C# 错误 CS0246 找不到类型或命名空间名称“Socket”(您是否缺少 using 指令或程序集引用)
- c - strtof - 返回错误的数字
- python - 如何使用装饰器修补 sys 属性?