我有一个与 Jersey 一起使用的简单 Flash 实现,如下所示:

@PostConstruct def before { flash.rotateIn }
@PreDestroy def after { flash.rotateOut }

object flash {
  val KeyNow  = "local.flash.now"
  val KeyNext = "local.flash.next"

  // case class Wrapper(wrapped: Map[String, Seq[String]])
  case class Wrapper(wrapped: String)

  def rotateIn {
    for {
      session <- Option(request.getSession(false))
      obj     <- Option(session.getAttribute(KeyNext))
    } {
      request.setAttribute(KeyNow, obj)
      session.removeAttribute(KeyNext)
    }
  }

  def rotateOut {
    for (obj <- Option(request.getAttribute(KeyNext))) {
      request.getSession.setAttribute(KeyNext, obj)
    }
  }

  def now = Option(request.getAttribute(KeyNow)) match {
    case Some(x: Wrapper) => x.wrapped
    case Some(x) if x.isInstanceOf[Wrapper] => "WHAT"
    case _ => "NOPE"
  }

  def next(value: String) {
    request.setAttribute(KeyNext, Wrapper(value))
  }
}

我在这里稍微简化了它,但它让我可以使用 flash.next 为 flash 设置一个值,并使用 flash.now 读取当前的 flash 值。

奇怪的是我的 now 值总是“WHAT”。如果我在 REPL 中做了类似的事情,我就不会遇到同样的问题:

val req = new org.springframework.mock.web.MockHttpServletRequest
val res = req.getSession
res.setAttribute("foo", Wrapper("foo"))
req.setAttribute("foo", res.getAttribute("foo"))
// Is not None
Option(req.getAttribute("foo")).collect { case x: Wrapper => x }

我错过了一些明显的东西吗?

编辑

我在 https://github.com/kardeiz/sc-issue-20160229 添加了一个最小的示例 webapp 来复制这个问题。

最佳答案

我试过你的例子。检查 my answer 的其他问题以了解模式匹配在这种情况下如何工作的详细信息。

简而言之,由于 Wrapper 是内部类,因此模式匹配还会检查“外部类”引用。似乎取决于应用程序服务器实现 Router.flash 可以是每个请求的不同实例,因此模式匹配失败。

对此的简单修复是使 Wrapper 顶级类,因此它没有对任何其他类的引用。

关于scala - servlet 代码中的类型信息丢失,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/35639950/

10-12 00:06
查看更多