假设我有一个代表角色的案例类,不同社交网络上的人.该类的实例是完全不可变的,并且保存在不可变集合中,最终由Akka actor修改.
现在,我有一个包含许多字段的case类,我收到一条消息,说我必须更新其中一个字段,如下所示:
case class Persona(serviceName : String,
serviceId : String,
sentMessages : Set[String])
// Somewhere deep in an actor
val newPersona = Persona(existingPersona.serviceName,
existingPersona.serviceId,
existingPersona.sentMessages + newMessage)
Run Code Online (Sandbox Code Playgroud)
注意我必须指定所有字段,即使只有一个更改.有没有办法克隆existingPersona并只替换一个字段,而不指定所有不更改的字段?我可以将其作为特征编写并将其用于我的所有案例类吗?
如果Persona是一个类似Map的实例,那么很容易做到.
说我有以下两个case class
es:
case class Address(street: String, city: String, state: String, zipCode: Int)
case class Person(firstName: String, lastName: String, address: Address)
Run Code Online (Sandbox Code Playgroud)
和以下Person
类的实例:
val raj = Person("Raj", "Shekhar", Address("M Gandhi Marg",
"Mumbai",
"Maharashtra",
411342))
Run Code Online (Sandbox Code Playgroud)
现在,如果我想更新zipCode
,raj
那么我将不得不做:
val updatedRaj = raj.copy(address = raj.address.copy(zipCode = raj.address.zipCode + 1))
Run Code Online (Sandbox Code Playgroud)
随着嵌套水平的提高,这将变得更加丑陋.是否有更清洁的方式(像Clojure的东西update-in
)来更新这样的嵌套结构?
这是对我上一个问题的答案的后续跟进.
假设我需要映射每个项目a:A
的List[A]
来b:B
使用功能def f(a:A, leftNeighbors:List[A]): B
和产生List[B]
.
显然,我不能只是map
在列表上调用,但我可以使用列表拉链.拉链是一个在列表中移动的光标.它提供对当前element(focus
)及其邻居的访问.
现在我可以替换我f
的 def f'(z:Zipper[A]):B = f(z.focus, z.left)
并将这个新函数传递f'
给cobind
方法Zipper[A]
.
这样的cobind
工作:它f'
用拉链调用,然后移动拉链,f'
用新的 "移动"拉链调用,再次移动拉链等等......直到拉链到达列表的末尾.
最后,cobind
返回一个新的拉链类型Zipper[B]
,可以将其转换为列表,从而解决问题.
现在请注意之间的对称性cobind[A](f:Zipper[A] => B):Zipper[B]
和bind[A](f:A => List[B]):List[B]
这就是为什么List
是Monad
和Zipper
是Comonad
.
是否有意义 ?