5

这可能是一种架构问题,但它仍然必须有一个“最佳实践解决方案”或一个公认的标准。

我说的是需要在网站上显示的某种静态数据,例如产品目录、菜单和菜单项列表、面包屑块列表等。我想在使用任何标准 CMS 时都可以使用此选项。

但我想为这个问题使用纯 JSF 解决方案。

所以,回到这个问题,我的阐述基于以下原则:

  1. 数据不应在 facelets 中硬编码,因此我使用数据库来保存值,如以下 db 脚本(在我的情况下为 MYSQL):

    CREATE TABLE CatalogueGroup (
        CatalogueGroupName VARCHAR(100) NOT NULL PRIMARY KEY,
        URLPath VARCHAR(200) NOT NULL,
        ParentGroupName VARCHAR(100) DEFAULT NULL,
        FOREIGN KEY (ParentGroupName) REFERENCES CatalogueGroup(CatalogueGroupName) ON UPDATE CASCADE ON DELETE SET NULL
    )ENGINE=InnoDB DEFAULT CHARSET=utf8;
    
  2. 然后,我想使用要保存在 @ManagedBean 中的实体类并将其显示在视图中,例如

    public class CatalogueGroup implements Serializable {
        private String catalogueGroupName;
        private List<CatalogueGroup> children = new ArrayList<CatalogueGroup>();
        private CatalogueGroup parentGroup;
        //other stuff of this bean
    }
    
    @ManagedBean
    @SessionScoped
    public class CatalogueBean implements Serializable {
        private CatalogueGroup catalogue;//loaded via CatalogueGroupDAO with condition parentGroup == null
        //other stuff of this bean
    }
    
    //snippet of xhtml view for a two-level catalogue
    <ul><h:outputText value="#{catalogueBean.catalogue.catalogueGroupName}" />
        <ui:rereat value="#{catalogueBean.catalogue.children}" var="group">
            <li><h:outputText value="#{group.catalogueGroupName}" /></li>
        </ui:repeat>
    </ul>
    

上述设置有效,但感觉很尴尬。因此,我想向 jsf 社区提出以下开放的“最佳实践”问题:

  1. 像这样设置目录 bean 的正确方法是什么:
    • @SessionScope将加载一次并在每个视图上重新显示的bean 或
    • @RequestScoped将在每个页面显示上访问数据库的bean。
  2. 有没有办法在 facelet 视图中设置递归函数,或者我应该将目录嵌套级别限制为 2 或 3。
  3. 我想显示某种修改后的目录,根据他的角色(在数据库表中添加的列)向登录用户公开更多组,并在没有用户登录时显示基本目录。此外,我想偶尔插入目录中的一些新组不会强制用户重新登录,而是立即重新显示正确的数据:
    • 我应该在业务层过滤组并将过滤后的 CatalogueGroup 暴露给 bean,或者我将加载整个目录并将其子项限制在视图中,并使用 render=false;
    • 在会话中公开整个目录是否正确?
    • 是否可以在将新数据库条目添加到服务器上所有当前处于活动状态的 CatalogueBean 时发送修改事件,强制刷新它们的属性(CatalogueGroup)或实现此功能,我只需要使用@RequestScopedbean;
    • 如果使用请求范围的 bean 是唯一的选择,那么明智的做法是访问数据库以经常获取数据,以至于很少更改或者有更智能的做事方式;
    • 当用户登录(和注销)时,已经有一个会话范围目录的实例,如何刷新它:我需要在动作/动作侦听器中手动执行它还是需要使会话无效或做一些更适合情况的事情.
4

2 回答 2

1

非常有趣,但也许如此开放的问题。

首先,范围取决于您将为目录提供的实用程序。@ViewScoped如果您想将数据链接到特定视图,或者@SessionScoped您的目标是实现购物篮之类的东西,我建议您进行检查。

对于递归函数,我认为您应该避免在纯视图(xhtml)层中进行这种做法,并使用PrimefacesRichfaces之类的库,它们具有内置组件来完成您想做的事情。使用它们,您只需在托管支持 bean 中以编程方式处理它们的逻辑结构。

最后,对于目录限制,我建议您仅从数据库中加载要使用的内容。这样,您就不会超载服务器-数据库服务器-客户端连接。您可以拥有一个@SessionScoped管理当前登录用户会话的 bean,并且根据该 bean,您可以向数据库询问某些值或其他值。

此外,您必须注意您的目录,如果您在会话期间对其进行大量修改,那么@ViewScopedbean 可能是更好的选择,因为每次请求视图时都会重新加载它。如果您为此使用@SessionScopedbean,则必须手动在其上添加每个更改,以便在会话期间保持更新。

"load the whole catalogue and limit its children in views with rendered=false"

如果你按照我说的方式去做,那是你不必做的工作。如果您正在管理一个复杂的树并在视图中引入更多逻辑,那么有条件地评估每个树节点可能会很麻烦。当然,您应该尽可能避免这种情况。

即使您已经找到了解决方案,这也是我的主要想法。

于 2013-01-22T09:15:04.920 回答
1

您可以使用缓存来存储各种用户角色的菜单。然后,当您在数据库中插入新数据(如果您从管理页面插入)时,您可以使缓存失效,或者您可以将缓存设置为在特定时间段(每天一次、每隔几个小时等)后过期,然后您缓存过期时再次读取数据。

于 2013-01-22T09:58:56.207 回答