2016-02-25 26 views
3

我有一個簡單的閃存實施與澤西使用看起來像這樣:損失類型信息的servlet代碼

@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.now讀取當前的閃存值。我的now值總是「什麼」。如果我做同樣的事情在我的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 } 

我失去了一些東西明顯?

編輯

我添加了一個小例子,web應用在https://github.com/kardeiz/sc-issue-20160229複製這個問題。

+0

'case Some(x):Option [Wrapper] => x.wrapped'工作嗎? (沒有測試它)。你的REPL會話與你的代碼不一樣(它只是測試'x'是'Wrapper',而不是'Some(x)'中的'x''是一個'Wrapper') –

+0

@TheArchetypalPaul那不會' t編譯,但是謝謝 –

+0

行。謝謝告訴我,對不起,沒有測試。 –

回答

2

我試過你的例子。請查詢my answer瞭解其他問題,以瞭解在這種情況下模式匹配的工作原理。

總之,因爲你的Wrapper是一個內部類,patter匹配還檢查「外部類」的引用。看來,根據應用服務器的實現,Router.flash對於每個請求可以是不同的實例,所以模式匹配失敗。

簡單的解決方法是製作Wrapper頂級類,所以它沒有任何其他類的引用。

+0

謝謝!我已經確認將'Wrapper'移出'Router類「可以解決這個問題。 –