4

我有一组模型对象和一组包装器对象来赋予它们额外的功能。

我希望能够使用允许您编写的相同速记简明地将模型对象的集合转换为包装器对象List("x", "y", "z").foreach(println),如下所示:

class Model
class ModelWrapper(val m: Model)
object ModelWrapper { def apply(model: Model) = new ModelWrapper(model) }

val m1 = new Model; val m2 = new Model; val m3 = new Model

List(m1, m2, m3).map(ModelWrapper)

因此ModelWrapper,作为参数传递的 被转换为ModelWrapper(_)对伴随对象的调用。

但是,当我尝试这个时,我得到一个类型不匹配的错误,如下所示:

<console>:14: error: type mismatch;
 found   : ModelWrapper.type (with underlying type object ModelWrapper)
 required: Model => ?
                  List(m1, m2, m3).map(ModelWrapper)

但是,如果我创建ModelWrapper一个case class, 并删除伴随对象,它就可以工作。我不想让它成为一个案例类,因为它添加的行为不适合案例类工作的整体方式。例如,具有相同模型类作为参数的两个包装类不一定相等。

我想知道的是,在这种情况下,案例类和伴随对象有什么区别?我可以在不使用案例类的情况下得到我想要的吗?

4

2 回答 2

10

你的伴生对象必须是一个函数:

object ModelWrapper extends Function1[Model, ModelWrapper] { def apply(model: Model) = new ModelWrapper(model) }

或者您可能更喜欢这个缩写:

object ModelWrapper extends (Model => ModelWrapper) { def apply(model: Model) = new ModelWrapper(model) }
于 2012-03-13T12:09:09.447 回答
4

出于某种原因,这些工作:

  List(m1, m2, m3).map(ModelWrapper(_))
  List(m1, m2, m3).map(ModelWrapper.apply)

似乎对于案例类,因为伴随对象是由编译器而不是你创建的,它知道你指的是ModelWrapper.apply. 当有同伴时,它认为您指的是同伴。

于 2012-03-13T12:58:33.920 回答