JSF内存泄漏:在同一页面上刷新/重定向时没有销毁bean



我对我的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,还需要跟踪整个视图本身。我不是百分之百确定客户端选项是如何工作的,但对我来说,让客户端跟踪"视图"而不仅仅是标识符是很危险的。

相关内容

  • 没有找到相关文章

最新更新