如何以及何时从JSF中的LRU缓存中删除视图范围bean?

时间:2013-12-23 08:05:22

标签: jsf jsf-2 view-scope

当视图范围bean被销毁时,我已经阅读了一个很好的答案。 (参见How and when is a @ViewScoped bean destroyed in JSF?)我自动假设已从视图范围缓存中删除了已销毁的bean。但是我可以看到bean仍然在缓存中,所以我想知道是否应该从LRU视图范围缓存中删除已销毁的视图范围bean,如果有的话?

在我们的应用程序中,我们打开sepeare选项卡/窗口中的所有详细信息。在一些打开/关闭(取决于numberOfViewsInSession)之后,我们可以看到ViewExpiredException,以防第一个细节窗口仍然打开并且用户打开和关闭另一个细节窗口,并且在一段时间后他想在第一个窗口中执行某些操作。我做了一些调试,我可以看到没有从LRU缓存中删除关闭的视图。

这是预期的行为还是我的代码中有问题?如果它是预期的行为,是否有任何有用的策略如何使用multitabs / multiwindow而没有由LRU缓存引起的大量ViewExpiredException?我知道我可以改变numberOfViewsInSession,但它是我想要使用的最后一个选择。

我准备了一个简单的测试用例,当我多次打开/关闭view.xhtml时,我可以看到LRUMap正在增长。

环境:JDK7,mojarra 2.2.4,tomcat 7.0.47

提前致谢

view.xhtml

    <!DOCTYPE html>
    <html xmlns="http://www.w3.org/1999/xhtml" xmlns:h="http://xmlns.jcp.org/jsf/html">
    <head>
        <title>View Bean</title>
        <meta charset="UTF-8"/>
    </head>
    <body>
        <h:form id="viewForm">
        <div>#{viewBean.text}</div>
        <h:commandButton id="closeButton" value="Close" action="/ClosePage.xhtml"/>
        </h:form>
    </body>
</html>

的index.xhtml

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
    <html xmlns="http://www.w3.org/1999/xhtml"
      xmlns:h="http://xmlns.jcp.org/jsf/html">
    <h:head>
        <title>Session bean</title>
    </h:head>
    <h:body>
        <h:form id="sessionForm">
            <h:outputText value="#{sessionBean.text}"/>
            <br/>
            <h:link id="linkView" value="Open view.xhmtl" outcome="/view.xhtml" target="_blank"/>
        </h:form>
    </h:body>
</html>

ClosePage.xhmtl

<!DOCTYPE html>
<html lang="en" xmlns="http://www.w3.org/1999/xhtml">               
    <head><title>Window will be closed</title></head>
    <body>
        <script>window.close();</script>
    </body>
</html>

ViewBean.java

package com.mycompany.mavenproject2;

import com.sun.faces.util.LRUMap;
import java.io.Serializable;
import java.util.Map;
import javax.annotation.PreDestroy;
import javax.faces.bean.ManagedBean;
import javax.faces.bean.ViewScoped;
import javax.faces.context.FacesContext;

@ManagedBean(name = "viewBean")
@ViewScoped
public class ViewBean implements Serializable {

    private static final long serialVersionUID = 13920902390329L;

    private int lruMapSize;

    /**
     * Creates a new instance of ViewBean
     */
    public ViewBean() {

        Map<String, Object> sessionMap = FacesContext.getCurrentInstance().getExternalContext().getSessionMap();
        LRUMap<String, LRUMap> lruMap = (LRUMap) sessionMap.get("com.sun.faces.renderkit.ServerSideStateHelper.LogicalViewMap");
        lruMapSize = lruMap == null ? 0 : lruMap.size();
    }

    @PreDestroy
    void destroyed() {
        System.out.println("View bean destroyed");
    }

    public String getText() {
        return "ViewBean LRU cache size:" + Integer.toString(lruMapSize);
    }

} 

SessionBean.java

package com.mycompany.mavenproject2;

import java.io.Serializable;
import javax.faces.bean.ManagedBean;
import javax.faces.bean.SessionScoped;

@ManagedBean(name = "sessionBean")
@SessionScoped
public class SessionBean implements Serializable {
    private static final long serialVersionUID = 1777489347L;

    /**
     * Creates a new instance of SessionBean
     */
    public SessionBean() {
    }

    public String getText() {
        return "Session bean text";
    }

}

1 个答案:

答案 0 :(得分:1)

我认为每个JSF开发人员最终都会遇到这种情况。真正的问题在于,您无法设计一个真正可靠的有状态系统,在该系统中,浏览器将向ViewScoped bean发送信号,告知它是通过页面完成的,从而允许支持bean自行销毁。这就是为什么JSF实现具有LRU高速缓存来限制会话使用的内存的原因,这对于日常应用程序来说是一个很好的解决方案。

在某些情况下,您知道自己完成了ViewScoped bean,例如来自该bean的重定向。对于这些,您可以编写自己的视图处理程序来执行更智能的缓存系统,但这不是一项简单的任务,坦率地说,不值得付出努力。

我提出的最简单的解决方案是使用javascript计时器在每个页面上使用ViewScoped bean执行ajax回发到服务器。 (将此计时器设置为每30秒执行一次似乎是合理的。)这会将与页面关联的ViewScoped bean移动到LRU缓存的底部,确保它们不会过期。

特别是,我使用一个primefaces poll组件将其拉出并粘贴在一个模板中供所有ViewScoped bean使用。通过将此组件放置为自己的形式,请求大小仍然很小。