Вложенные JSF-составные компоненты, приводящие к исключению переполнения стека

Эта проблема

Когда я пытаюсь вложить Composite Component внутри себя, с некоторой логикой, чтобы завершить бесконечную рекурсию, я получаю исключение переполнения стека. Я понимаю, что <c:xxx> теги выполняются во время сборки представления, поэтому я не ожидал, что будет построение бесконечного представления, как я полагаю, имело место.

Это составной компонент simpleNestable.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:composite="http://java.sun.com/jsf/composite"
  xmlns:h="http://java.sun.com/jsf/html"
  xmlns:em="http://xmlns.jcp.org/jsf/composite/emcomp"

  xmlns:c="http://xmlns.jcp.org/jsp/jstl/core">

    <h:head>
        <title>This content will not be displayed</title>
    </h:head>
    <h:body>
        <composite:interface>
            <composite:attribute name="depth" required="true" type="java.lang.Integer"/>
        </composite:interface>

        <composite:implementation>
            <c:if test="#{cc.attrs.depth lt 3}">
                 #{cc.attrs.depth}
                 #{cc.attrs.depth+1}
                 <em:simpleNestable depth="#{cc.attrs.depth+1}" /> 

            </c:if>

        </composite:implementation>
    </h:body>
</html>

Вот как это используется

<h:head>
    <title>Facelet Title</title>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
    <h:outputStylesheet name="./css/default.css"/>
    <h:outputStylesheet name="./css/cssLayout.css"/>
</h:head>
<h:body>        
     <emcomp:simpleNestable depth="1"/>

</h:body>

Исключение переполнения стека

java.lang.StackruError
    at com.sun.faces.facelets.el.TagValueExpression.getValue(TagValueExpression.java:109)
    at javax.faces.component.UIComponentBase$AttributesMap.get(UIComponentBase.java:2407)
    at com.sun.faces.el.CompositeComponentAttributesELResolver$ExpressionEvalMap.get(CompositeComponentAttributesELResolver.java:393)
    at javax.el.MapELResolver.getValue(MapELResolver.java:199)
    at com.sun.faces.el.DemuxCompositeELResolver._getValue(DemuxCompositeELResolver.java:176)
    at com.sun.faces.el.DemuxCompositeELResolver.getValue(DemuxCompositeELResolver.java:203)
    at com.sun.el.parser.AstValue.getValue(AstValue.java:140)
    at com.sun.el.parser.AstValue.getValue(AstValue.java:204)
    at com.sun.el.parser.AstPlus.getValue(AstPlus.java:60)
    at com.sun.el.ValueExpressionImpl.getValue(ValueExpressionImpl.java:226)
    at org.jboss.weld.el.WeldValueExpression.getValue(WeldValueExpression.java:50)
    at com.sun.faces.facelets.el.ContextualCompositeValueExpression.getValue(ContextualCompositeValueExpression.java:158)
    at com.sun.faces.facelets.el.TagValueExpression.getValue(TagValueExpression.java:109)
    at javax.faces.component.UIComponentBase$AttributesMap.get(UIComponentBase.java:2407)
    at com.sun.faces.el.CompositeComponentAttributesELResolver$ExpressionEvalMap.get(CompositeComponentAttributesELResolver.java:393)
    at javax.el.MapELResolver.getValue(MapELResolver.java:199)
    at com.sun.faces.el.DemuxCompositeELResolver._getValue(DemuxCompositeELResolver.java:176)
    at com.sun.faces.el.DemuxCompositeELResolver.getValue(DemuxCompositeELResolver.java:203)
    at com.sun.el.parser.AstValue.getValue(AstValue.java:140)
    at com.sun.el.parser.AstValue.getValue(AstValue.java:204)
    at com.sun.el.parser.AstPlus.getValue(AstPlus.java:60)

Вопрос

Как я могу вкладывать составные компоненты (или аналогичные) в себя (на неопределенную глубину) без получения исключения переполнения стека

Почему я хочу это

У меня есть произвольно вложенные данные, которые я хочу представить в рамках вложенного collapsibleSubTable из RichFaces, альтернативы моему подходу очень приветствуются

1 ответ

Решение

Проблема была в контексте #{cc} и состояние составного атрибута. #{cc} в любом атрибуте вложенных составных ссылок сама, а не родительская. Атрибут с состоянием означает, что #{cc} был переоценен у каждого ребенка, который в свою очередь ссылается на себя, а не на родителя. Отсюда переполнение стека. Он оценивает глубину себя в бесконечном цикле.

Я обманул состояние атрибута, сделав его без сохранения состояния, используя вспомогательный компонент, как показано ниже, который немедленно оценивает его и назначает его как свойство компонента:

@FacesComponent("treeComposite")
public class TreeComposite extends UINamingContainer {

    private Integer depth;

    @Override
    public void setValueExpression(String name, ValueExpression binding) {
        if ("depth".equals(name)) {
            setDepth((Integer) binding.getValue(getFacesContext().getELContext()));
        }
        else {
            super.setValueExpression(name, binding);
        }
    }

    public Integer getDepth() {
        return depth;
    }

    public void setDepth(Integer depth) {
        this.depth = depth;
    }

}

Который должен быть объявлен в интерфейсе componentType как показано ниже:

<cc:interface componentType="treeComposite">
    <cc:attribute name="depth" type="java.lang.Integer" />
</cc:interface>

И в реализации вы должны в тесте ссылаться на свойство без сохранения состояния, а во вложенной составной ссылке - на родительское (потому что #{cc} в атрибуте вложенного композита ссылки на сам вложенный композит):

<cc:implementation>
    <br />We're at depth #{cc.depth}.
    <c:if test="#{cc.depth gt 0}">
        <my:tree depth="#{cc.parent.depth - 1}" />
    </c:if>
</cc:implementation>

Я только изменил значение "глубина" здесь, чтобы оно было наоборот, чтобы оно было просто декларативным от клиента без необходимости редактировать его в реализации. Итак, в клиенте вы должны сказать depth="#{3}" если вы хотите 3 вложенных детей:

<my:tree depth="#{3}" />

Обратите внимание на важность того, чтобы оно было выражением EL, а не литералом. Иначе setValueExpression() в резервном компоненте не будет вызван.

Другие вопросы по тегам