如何从其他包引用scala Enumeration

时间:2012-04-19 11:40:31

标签: scala playframework-2.0

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

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

enum(UserStatus)

有时它抱怨无法找到枚举

not found: type UserStatus

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

enum的生成源似乎没问题,并且它与我生活在同一个地方,使用相同的另一个enum很有效...

任何建议?

更多信息

枚举的生成源是:

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)
  )
}

1 个答案:

答案 0 :(得分:8)

您的问题可能来自以下事实:在Scala中,您可以对类型使用相同的标识符

撰写以下内容时:

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 类型。< / p>