以可堆叠特征模式覆盖抽象类型成员

时间:2013-05-29 00:01:03

标签: scala traits abstract-type

假设我们想要定义如何在某些数据上累积结果的方法:

case class Data(x: Int, y: Int)

我们定义了一个特征:

trait Accumulator {
  type R   
  def add(acc: R, d: Data): R
  def zero: R
}

一个简单的实现:

trait XAccumulator extends Accumulator {
  type R = Int
  def add(acc: Int, d: Data) = acc + d.x
  def zero = 0
}

我想使用stackable-trait模式来使用多个这些简单的累加器:

trait TraceYAccumulator extends Accumulator {
  abstract override type R = (Seq[Int], super.R)
  // fails:
  // `abstract override' modifier not allowed for type members

  def add(acc: R, d: Data) = {
    val r = super.add(acc._2, d)
    (acc._1 :+ d.y, r)
  }

  def zero = (Seq.empty[Int], super.zero)
}

显然我不允许覆盖抽象类型的成员。如何使用可堆叠特征模式更改重写方法的结果类型?

我的第二种方法是使用类型参数:

trait Accumulator[R] {
  def add(acc: R, d: Data): R
  def zero: R
}

trait XAccumulator extends Accumulator[Int] {
  def add(acc: Int, d: Data) = acc + d.x
  def zero = 0
}

但现在变得非常奇怪:

trait TraceYAccumulator[T] extends Accumulator[(Seq[Int], T)] {
  this: Accumulator[T] =>

  def add(acc: (Seq[Int], T), d: Data): (Seq[Int], T) = {
    val r = this.add(acc._2, d)
    // fails: overloaded method value add with alternatives:
    // (acc: (Seq[Int], T),d: Data)(Seq[Int], T) <and>
    // (acc: _14695,d: Data)_14695 cannot be applied to (T, Data)
    (acc._1 :+ d.y, r)
  }
  def zero: (Seq[Int], T) = (Seq.empty[Int], this.zero)
}

由于超类和混合类具有相同的方法名称(显然),我们无法引用正确的方法。我唯一的选择是使用作文吗?

如何堆叠此类操作?

1 个答案:

答案 0 :(得分:1)

第一种方法有一个基本的理论问题:

假设我们被允许写这些特征。然后考虑以下事项:

val acc = new XAccumulator with TraceYAccumulator
val xacc = acc: XAccumulator

由于xaccXAccumulator,我们知道x.R =:= Int。糟糕!

第二种方法存在实施问题:您不能从同一特征的不同实例继承。

illegal inheritance; anonymous class $anon inherits different type instances of trait Accumulator: Accumulator[Int] and Accumulator[(Seq[Int], Int)]

因此,组合(或者某些类型类hackery)似乎是唯一的选择。