1

我是 JSF 编程的新手,并且设计了一个相当简单的 JSF 2.0 Web 应用程序。它具有丰富的:左侧的树,用作分层菜单。当用户单击树中的某些内容时,将触发选择处理程序并触发页面的部分重新加载,该页面显示与您刚刚单击的内容相关的内容。

问题是界面感觉迟缓,所以我启用了一些日志记录并添加了一个自定义阶段侦听器,让我了解瓶颈在哪里。

2012-05-21 07:58:05.516 DEBUG NodeBean - Retrieving properties
2012-05-21 07:58:05.516 DEBUG CustomPhaseListener - Before phase: APPLY_REQUEST_VALUES 2
2012-05-21 07:58:05.876 DEBUG RepositoryBean - Selection change event invoked (size: 1)
2012-05-21 07:58:05.876 DEBUG RepositoryBean - Selection change event handled
2012-05-21 07:58:05.876 DEBUG CustomPhaseListener - After phase: APPLY_REQUEST_VALUES 2
2012-05-21 07:58:05.876 DEBUG CustomPhaseListener - Before phase: PROCESS_VALIDATIONS 3
2012-05-21 07:58:06.469 DEBUG CustomPhaseListener - After phase: PROCESS_VALIDATIONS 3
2012-05-21 07:58:06.469 DEBUG CustomPhaseListener - Before phase: UPDATE_MODEL_VALUES 4
2012-05-21 07:58:06.844 DEBUG CustomPhaseListener - After phase: UPDATE_MODEL_VALUES 4
2012-05-21 07:58:06.844 DEBUG CustomPhaseListener - Before phase: INVOKE_APPLICATION 5
2012-05-21 07:58:06.844 DEBUG CustomPhaseListener - After phase: INVOKE_APPLICATION 5
2012-05-21 07:58:06.860 DEBUG CustomPhaseListener - Before phase: RENDER_RESPONSE 6
2012-05-21 07:58:06.985 DEBUG NodeBean - Retrieving properties
2012-05-21 07:58:07.001 DEBUG RepositoryBean - Retrieving content
2012-05-21 07:58:07.001 DEBUG RepositoryBean - Content retrieved
2012-05-21 07:58:07.376 DEBUG CustomPhaseListener - After phase: RENDER_RESPONSE 6

NodeBean 是一个请求范围的 bean,它有一个“getProperties()”方法。我不知道为什么一开始就调用它,但它的开销很小,所以这是另一个问题。

正如您所见,JSF 生命周期需要很长时间才能完成,而自定义代码在开销方面相当微不足道。经过一番谷歌搜索,我发现“immediate='true'”可以让你跳过一些步骤,但将它添加到 rich:tree 没有任何效果。我认为我做错了什么,但对于我的生活,我无法弄清楚是什么。

附带说明:有时(通常在用户不活动之后)该过程将在阶段 3 之后停止,这意味着没有内容返回给用户。在用户重新加载页面之前,rich:tree 中的任何选择都将具有相同的行为。

编辑:

经过一些进一步的调试后,richfaces 树似乎在每个请求上都会重建几次。它与这个问题有关:JSF2 Richfaces 4.1.0 Ajax partial rendering of tree

我的用例是:树基本上是静态的,如果您选择一个项目,与该项目相关的属性和内容应该出现在右侧。所以基本上我有一个 selectionchange 侦听器,它注册您选择的内容,并重新呈现使用该设置加载属性和内容的中央组件。树应该(几乎)永远不会被刷新。但实际发生的情况是,即使部分重新渲染没有提及或引用树,也会为阶段 2、3、4 和 6 重建整个树。

最初我有这个:

<rich:tree id="tree" var="node"
        value="#{repository.browser}" 
        render="node" 
        toggleType="client" selectionType="ajax"

如果您将“selectionType”设置为客户端,问题就会消失,但不会在后端重新渲染或调用任何内容。我尝试添加类似

<f:ajax event="click" render=":node"/>

但这也失败了(树仍然在后台重建)。有没有办法缓存树或者干脆不为每个请求重建它?

4

1 回答 1

1

我已经为我的特定用例解决了这个问题,但我不确定它是否适用于其他人。我更新了 org.richfaces.component.TreeRange.shouldIterateChildren() 方法:

public boolean shouldIterateChildren() {
    if (tree.getRowKey() == null) {
        return true;
    }

    if (tree.isLeaf()) {
        return false;
    }

    return traverseAll || tree.isExpanded();
}

至:

public boolean shouldIterateChildren() {
    if (tree.isLeaf())
        return false;
    else {
        char separatorChar = UINamingContainer.getSeparatorChar(FacesContext.getCurrentInstance());
        String clientId = tree.getClientId();
        boolean render = false;
        for (String idToRender : FacesContext.getCurrentInstance().getPartialViewContext().getRenderIds()) {
            // render the tree if you explicitly mention either the client id (e.g. "menuForm:tree") or the parent component client id (e.g. "menuForm")
            // note that when clicking on an object in the tree, the following render target is requested: menuForm:tree@selection
            if (clientId.equals(idToRender) || clientId.matches(idToRender + separatorChar + ".*")) {
                render = true;
                break;
            }
        }
        // always render if it's not a postback
        return render || !FacesContext.getCurrentInstance().isPostback();
    }
}

我仍然不知道我是否使用了rich:tree 错误或者它实际上是一个错误,我已经向 jboss-users 邮件列表发送了一封邮件,如果它实际上是一个错误,将创建一个 jira 问题。至少现在我的 Web 应用程序即使使用大树也能快速响应。

于 2012-05-22T13:39:50.083 回答