我对我的managedbean使用JSF视图。我在html视图中调用托管bean。
刷新页面时,允许一个新的托管bean,而旧的bean保留在内存中。如果我刷新,刷新,刷新…我在内存中有大量的托管bean。
我试着运行垃圾回收程序,但没有任何结果。托管bean一直保留到会话过期。在会话结束时,通过用户操作或超时,托管bean将被释放。
我只有在停留在同一页面时才会遇到这个问题(在浏览器上刷新- f5 -或没有JSP标记或java代码重定向的重定向链接)。当我更改页面时,只有一个托管bean被释放(最后一个)。其他的留在记忆里。它似乎与JSP标记一起工作,托管bean在重定向后被释放。
第二个Bug:有一次,我来到一个奇怪的地方。我有一些管理bean没有在内存中释放,我使用JSP标签重定向(可能3/4次)和系统做预销毁/PostConstruct/预销毁……在此之后,每个重定向做PostConstruct/Predestroy不JSP标签或java重定向,预销毁/PostConstruct/预销毁JSP标签。在内存中,未释放的托管bean随后被释放。我真的不知道到底是什么情况,复制这个,但它做了很多次。
有人有解决这个问题的办法吗?我看到一个帖子看起来像问题在这里,我报告一个错误在java.net配置:Java EE 7/glassfish 4
AViewscope.java:
package com.btm.viewscopetest;
import java.io.IOException;
import java.io.Serializable;
import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;
import javax.faces.bean.ManagedBean;
import javax.faces.bean.ViewScoped;
import javax.faces.context.ExternalContext;
import javax.faces.context.FacesContext;
@ManagedBean
@ViewScoped
public class AViewscope implements Serializable {
@PostConstruct
public void postConstruct() {
System.out.println("PostConstruct");
}
@PreDestroy
public void preDestroy() {
System.out.println("PreDestroy");
}
public AViewscope() {
}
public String getSomething() {
return "Something";
}
public void redirect() throws IOException {
FacesContext facesContext = FacesContext.getCurrentInstance();
ExternalContext externalContext = facesContext.getExternalContext();
externalContext.redirect("apage.xhtml");
}
}
apage.xhtml:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:h="http://xmlns.jcp.org/jsf/html">
<head>
<title>A page</title>
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
</head>
<body>
<h:form>
<h:outputText value="Get something: #{aViewscope.something}" />
<br/>
<br/>
<h:commandButton value="Stay on this page with java " actionListener="#{aViewscope.redirect}"/>
<br/>
<h:commandButton value="Stay on this page with direct link" action="apage" />
<br/>
<h:commandButton value="Go to another page" action="anotherPage"/>
<br/>
<h:commandLink value="Stay on this page with direct link h tag " action="apage.xhtml" />
<a href="apage.xhtml">Stay on this page with direct link a tag</a>
</h:form>
</body>
</html>
anotherPage.xhtml:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:h="http://xmlns.jcp.org/jsf/html">
<head>
<title>Another Page</title>
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
</head>
<body>
<h:form>
<h:commandButton value="Go to another page" action="apage"/>
</h:form>
</body>
</html>
web . xml:
<?xml version="1.0" encoding="UTF-8"?>
<web-app version="3.1" xmlns="http://xmlns.jcp.org/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd">
<context-param>
<param-name>javax.faces.PROJECT_STAGE</param-name>
<param-value>Development</param-value>
</context-param>
<servlet>
<servlet-name>Faces Servlet</servlet-name>
<servlet-class>javax.faces.webapp.FacesServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>Faces Servlet</servlet-name>
<url-pattern>*.xhtml</url-pattern>
</servlet-mapping>
<session-config>
<session-timeout>
30
</session-timeout>
</session-config>
<welcome-file-list>
<welcome-file>index.xhtml</welcome-file>
</welcome-file-list>
</web-app>
我在内存中有大量的托管bean。我尝试运行垃圾回收器,没有附加
这不是一个bug,这是故意的。如果在web.xml
中将javax.faces.STATE_SAVING_METHOD
设置为server
,那么JSF将在内存中维护一定数量的bean。客户机现在将始终提交一个javax.faces.ViewState
(id),从而允许服务器识别要加载的正确bean。
您可以通过将上下文参数com.sun.faces.numberOfViewsInSession
设置为符合您需要的数字来限制每个会话存储的视图数量。
但是要注意,如果您选择1
,并且用户使用向后导航,如果bean已被删除,他将遇到ViewExpired
异常。
减少内存负载的另一个选项是通过将javax.faces.STATE_SAVING_METHOD
设置为client
,将视图存储处理到客户端。然而,这将导致网络流量的增加,因为现在客户端不仅需要跟踪javax.faces.ViewState
,还需要跟踪整个视图本身。我不是百分之百确定客户端选项是如何工作的,但对我来说,让客户端跟踪"视图"而不仅仅是标识符是很危险的。