8

我想创建一个可变的协变类,所以我需要添加一个绑定到 setter 方法的较低类型。但是我也希望setter方法设置一个字段,所以我猜这个字段需要绑定相同的类型?

class Thing[+F](initialValue: F) {

    private[this] var secondValue: Option[G >: F] = None

    def setSecondValue[G >: F](v: G) = {
        this.secondValue = Some(v)
     }
}

该方法编译得很好。但是名为 secondValue 的字段根本无法编译,并显示错误消息:

    Multiple markers at this line
        - ']' expected but '>:' found.
        - not found: type G

我需要做什么?

4

2 回答 2

11

@mhs 的答案是正确的。

您还可以使用通配符语法(如在 java 中),其含义完全相同:

scala> :paste
// Entering paste mode (ctrl-D to finish)

class Thing[+F](initialValue: F) {
  private[this] var secondValue: Option[_ >: F] = None

  def setSecondValue[G >: F](v: G) = {
    this.secondValue = Some(v)
  }

  def printSecondValue() = println(secondValue)
}

// Exiting paste mode, now interpreting.

defined class Thing

scala> val t = new Thing(Vector("first"))
t: Thing[scala.collection.immutable.Vector[java.lang.String]] = Thing@1099257

scala> t.printSecondValue()
None

scala> t.setSecondValue(Seq("second"))

scala> t.printSecondValue()
Some(List(second))
于 2012-08-08T07:02:34.990 回答
9

您需要forSome构造,它G作为存在类型引入:

class Thing[+F](initialValue: F) {
  private[this] var secondValue: Option[G] forSome { type G >: F} = None

  def setSecondValue[G >: F](v: G) = {
    this.secondValue = Some(v)
  }
}

在您的原始代码中,secondValue,G已被无中生有,即没有正确引入。如果setSecondValue用户(或编译器)G在调用站点绑定,但对于一个不是选项的字段(特别是因为你的字段是私有的)。在此处此处此处阅读有关forSomeScala 中存在类型的更多信息。

于 2012-08-08T06:42:55.943 回答