这可能是一种架构问题,但它仍然必须有一个“最佳实践解决方案”或一个公认的标准。
我说的是需要在网站上显示的某种静态数据,例如产品目录、菜单和菜单项列表、面包屑块列表等。我想在使用任何标准 CMS 时都可以使用此选项。
但我想为这个问题使用纯 JSF 解决方案。
所以,回到这个问题,我的阐述基于以下原则:
数据不应在 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;
然后,我想使用要保存在 @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 社区提出以下开放的“最佳实践”问题:
- 像这样设置目录 bean 的正确方法是什么:
@SessionScope
将加载一次并在每个视图上重新显示的bean 或@RequestScoped
将在每个页面显示上访问数据库的bean。
- 有没有办法在 facelet 视图中设置递归函数,或者我应该将目录嵌套级别限制为 2 或 3。
- 我想显示某种修改后的目录,根据他的角色(在数据库表中添加的列)向登录用户公开更多组,并在没有用户登录时显示基本目录。此外,我想偶尔插入目录中的一些新组不会强制用户重新登录,而是立即重新显示正确的数据:
- 我应该在业务层过滤组并将过滤后的 CatalogueGroup 暴露给 bean,或者我将加载整个目录并将其子项限制在视图中,并使用 render=false;
- 在会话中公开整个目录是否正确?
- 是否可以在将新数据库条目添加到服务器上所有当前处于活动状态的 CatalogueBean 时发送修改事件,强制刷新它们的属性(CatalogueGroup)或实现此功能,我只需要使用
@RequestScoped
bean; - 如果使用请求范围的 bean 是唯一的选择,那么明智的做法是访问数据库以经常获取数据,以至于很少更改或者有更智能的做事方式;
- 当用户登录(和注销)时,已经有一个会话范围目录的实例,如何刷新它:我需要在动作/动作侦听器中手动执行它还是需要使会话无效或做一些更适合情况的事情.