当视图范围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";
}
}
答案 0 :(得分:1)
我认为每个JSF开发人员最终都会遇到这种情况。真正的问题在于,您无法设计一个真正可靠的有状态系统,在该系统中,浏览器将向ViewScoped bean发送信号,告知它是通过页面完成的,从而允许支持bean自行销毁。这就是为什么JSF实现具有LRU高速缓存来限制会话使用的内存的原因,这对于日常应用程序来说是一个很好的解决方案。
在某些情况下,您知道自己完成了ViewScoped bean,例如来自该bean的重定向。对于这些,您可以编写自己的视图处理程序来执行更智能的缓存系统,但这不是一项简单的任务,坦率地说,不值得付出努力。
我提出的最简单的解决方案是使用javascript计时器在每个页面上使用ViewScoped bean执行ajax回发到服务器。 (将此计时器设置为每30秒执行一次似乎是合理的。)这会将与页面关联的ViewScoped bean移动到LRU缓存的底部,确保它们不会过期。
特别是,我使用一个primefaces poll组件将其拉出并粘贴在一个模板中供所有ViewScoped bean使用。通过将此组件放置为自己的形式,请求大小仍然很小。