4

我无法理解为什么我不能引用 scala 枚举类型。

问题是有时我可以引用枚举:

enum(UserStatus)

有时它抱怨无法找到枚举

not found: type UserStatus

为什么我有时不能引用 Enumeration 类?

枚举的生成源似乎没问题,并且它与我拥有的另一个枚举一起工作得很好,它住在同一个地方,同样的用法......

有什么建议么?

更多信息

枚举的生成源是:

public final class models.UserStatus extends java.lang.Object{
    public static final scala.Enumeration$Value Busy();
    public static final scala.Enumeration$Value Free();
    public static final scala.Enumeration$ValueSet$ ValueSet();
    public static final scala.Enumeration$Value withName(java.lang.String);
    public static final scala.Enumeration$Value apply(int);
    public static final int maxId();
    public static final scala.Enumeration$ValueSet values();
    public static final java.lang.String toString();
}

我正在尝试为 play framework 2.0 实现枚举映射器

def enumFormat[E <: Enumeration](enum: E): Formatter[E#Value] = new Formatter[E#Value] {
  def bind(key: String, data: Map[String, String]) = {
    Formats.stringFormat.bind(key, data).right.flatMap { s =>
      scala.util.control.Exception.allCatch[E#Value]
        .either(enum.withName(s))
        .left.map(e => Seq(FormError(key, "error.enum", Nil)))
    }
  }
  def unbind(key: String, value: E#Value) = Map(key -> value.toString)
}

而这个调用映射器的方法

def enum[E <: Enumeration](enum: E): Mapping[E#Value] = of(enumFormat(enum))

这意味着映射器在使用表单绑定时会自动在枚举之间进行转换

使用伪代码。

package models {
  object UserStatus extends Enumeration {
    val Free = Value("free")
    val Busy = Value("busy")
  }

  case class User(
    status: UserStatus.Value = UserStatus.Free
  )
}

package controllers {
  imports models._
  val userForm = Form(
    mapping(
      "status" -> enum(UserStatus)
    )(User.apply)(User.unapply)
  )
}
4

1 回答 1

10

您的问题可能来自这样一个事实,即在 Scala 中您可以为typevalue使用相同的标识符

当您编写以下内容时:

object Foo extends Enumeration {
  val A, B, C = Value
}

你定义了一个Foo 对象,但你没有定义一个Foo 类型。如果您来自 Java 背景,您可能会觉得这种行为并不直观。

那么Foo枚举值的类型是什么?它们的类型是Foo.Value(称为路径依赖类型)。

如果要定义Foo与枚举值类型对应的类型,可以给该Foo.Value类型起别名:

object Foo extends Enumeration {
  type Foo = Value    // Type alias
  val A, B, C = Value
}

现在您可以通过编写来引用枚举类型Foo.Foo。您可以导入Foo对象的字段以减少语法开销:

import Foo._
def bar(foo: Foo): String = foo match {
  case A => "value A"
  case B => "value B"
  case C => "value C"
}

在上面的代码中,第一行指的是Foo 对象,第二行指的是Foo 类型

于 2012-04-19T21:03:01.843 回答