Утечка памяти JSF: bean-компонент не уничтожается при обновлении/перенаправлении на той же странице

Я использую JSF viewscope для своего управляемого компонента. Я вызываю управляемый компонент в виде html.

Когда страница обновляется, разрешен новый управляемый компонент, а старый остается в памяти. Если я обновляю, обновляю и обновляю... У меня огромное количество управляемых bean-компонентов в памяти.
Я пытаюсь запустить сборщик мусора, ничего не добавляется. Управляемые компоненты остаются до истечения срока действия сеанса. В конце сеанса по действию пользователя или по тайм-ауту управляемый компонент освобождается.

У меня есть эта проблема только тогда, когда я остаюсь на той же странице (обновление в браузере -F5- или ссылка перенаправления без тега JSP или перенаправления кода Java). Когда я меняю страницу, освобождается только один управляемый компонент (последний). Остальные остаются в памяти. При работе с тегом JSP управляемый компонент освобождается после перенаправления.

Вторая ошибка: некоторое время я столкнулся со странным случаем: у меня есть управляемый компонент, не освобожденный в памяти, я использую перенаправление тега JSP (может быть 3/4 раза), и система выполняет Predestroy/PostConstruct/Predestroy... И после этого каждое перенаправление сделать PostConstruct/Predestroy для тега не JSP или перенаправления Java, Predestroy/PostConstruct/Predestroy для тега JSP. После этого в памяти освобождается не освобожденный управляемый компонент. Я действительно не знаю, что на самом деле происходит, это воспроизводится, но это происходит несколько раз.

Есть ли у кого-нибудь решение этой проблемы? Я вижу сообщение, похожее на проблему здесь и я сообщаю об ошибке на java.net

Конфигурация: Java EE 7/стеклянная рыба 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>

другая страница.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>

веб.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>

person user3805731    schedule 24.07.2015    source источник
comment
Дубликат: stackoverflow.com/questions/20930396/ (и на SO есть более похожие сообщения, пожалуйста, найдите их и ссылайтесь на них   -  person Kukeltje    schedule 24.07.2015
comment
Я уже вижу этот пост (я даю ссылку в своем вопросе). На самом деле это не говорит мне, как исправить мою проблему... И, к сожалению, я не могу использовать омнифейсы в своем проекте.   -  person user3805731    schedule 24.07.2015
comment
Извините, из-за всего текста перед этим, перестал читать, так как вспомнил эту ссылку. В следующий раз лучше опубликовать их ранее в своем вопросе.   -  person Kukeltje    schedule 24.07.2015
comment
в порядке. У вас есть идея, чтобы решить эту проблему?   -  person user3805731    schedule 24.07.2015
comment
Ну, на SO есть еще посты. Некоторые связанные представления в сеансе. Так что, возможно, они не выпущены, как указано в ответе   -  person Kukeltje    schedule 24.07.2015
comment
Утечка памяти? Вы подразумеваете, что bean-компонент не уничтожается при уничтожении связанного HTTP-сеанса?   -  person BalusC    schedule 27.07.2015
comment
@BalusC Bean-компоненты уничтожаются в конце сеанса HTTP.   -  person user3805731    schedule 28.07.2015
comment
Тогда нет утечки памяти.   -  person BalusC    schedule 28.07.2015
comment
@BalusC ну хорошо... Ты прав! Проблема цикла жизни, если вы предпочитаете... Любое решение?   -  person user3805731    schedule 28.07.2015


Ответы (1)


У меня в памяти огромное количество управляемых бинов. Я пытаюсь запустить сборщик мусора, ничего не добавляется

Это не ошибка, это сделано намеренно. JSF поддерживает определенное количество bean-компонентов в памяти, если для вашего javax.faces.STATE_SAVING_METHOD установлено значение server в файле web.xml. Теперь клиент всегда будет отправлять javax.faces.ViewState (id), позволяя серверу определить, какие bean-компоненты нужно загрузить.

Вы можете ограничить количество просмотров, сохраняемых за сеанс, установив параметр контекста com.sun.faces.numberOfViewsInSession на число, соответствующее вашим потребностям.

Но имейте в виду, что если вы выберете 1, а пользователь будет использовать обратную навигацию, он столкнется с ViewExpired исключениями, если компонент был удален.

Другим вариантом уменьшения нагрузки на память может быть передача представления-хранилища клиенту, установив javax.faces.STATE_SAVING_METHOD в client. Однако это приведет к увеличению сетевого трафика, поскольку теперь клиенту нужно иметь не только javax.faces.ViewState для отслеживания, но и само представление целиком. Я не уверен на сто процентов, как именно работает клиентская опция, но мне кажется опасным позволять клиенту отслеживать «Просмотр», а не только идентификатор.

person dognose    schedule 24.07.2015
comment
Как вы можете сказать, что это преднамеренно? (не саркастический, это реальный вопрос) Если это сделано намеренно, то почему, когда я использую тег JSP, управляемый компонент освобождается? И почему, когда я нахожусь в обычной навигации (от страницы к другой странице), управляемый компонент свободен? Я понимаю, что принцип повторного использования удалось не потерять время выделения памяти (память пула), но для меня это не так. Я пробую javax.faces.STATE_SAVING_METHOD и com.sun.faces.numberOfViewsInSession , я пробую и повторяю этот метод, который не работает. Вы пробуете это? - person user3805731; 24.07.2015