servlet代码中类型信息的丢失

雅各布·布朗

我有一个用于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.nextflash的值并使用来读取当前的flash值flash.now

奇怪的是,我的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示例来复制此问题

艾文

我试过你的例子。检查我的答案是否有其他问题,以了解在这种情况下模式匹配的工作原理。

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

一个简单的解决方法是制作Wrapper顶级类,因此它没有引用任何其他类。

本文收集自互联网,转载请注明来源。

如有侵权,请联系 [email protected] 删除。

编辑于
0

我来说两句

0 条评论
登录 后参与评论

相关文章