我有以下XHTML:
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:ui="http://java.sun.com/jsf/facelets"
xmlns:h="http://java.sun.com/jsf/html"
xmlns:f="http://java.sun.com/jsf/core">
<head>
<title>TODO supply a title</title>
</head>
<body>
<f:metadata>
<f:viewParam id="productCV" name="productName" value="#{productBean.product}"
converter="#{productConverter}" required="true"/>
</f:metadata>
<ui:composition template="/templates/mastertemplate.xhtml">
<!-- Define the page title for this page-->
<ui:define name="pageTitle">
<h:outputFormat value="#{msgs.productPageTitle}">
<f:param value="#{productBean.product.description}"/>
</h:outputFormat>
</ui:define>
<!-- Pass the categoryName parameter to the sidebar so the category of this product is highlighted-->
<ui:param name="categoryName" value="#{productBean.product.categoryName}"/>
<ui:define name="content">
<!-- If productconversion failed, show this error-->
<h:message id="error" for="productCV" style="color: #0081c2;" rendered="#{productBean.product == null}" />
<!-- If productconversion succeeded show the product page-->
<h:panelGroup rendered="#{productBean.product != null}">
<p>#{productBean.product.description} #{productBean.product.categoryName}</p>
<h:form>
<h:commandLink action="#{cartBean.addItemToCart(productBean.product)}">
<f:ajax event="action" render=":cart :cartPrice" />
<h:graphicImage value="resources/img/addToCart.gif"/>
</h:commandLink>
</h:form>
</h:panelGroup>
</ui:define>
</ui:composition>
</body>
</html>
在顶部,我接受一个字符串作为GET参数,我通过一个转换器运行,然后得到一个Product
对象,我把它放在productBean.product
中,该bean有Product
属性的setter和getter,这就是全部。
然后我使用这个对象来显示信息等,这很好。我还添加了commandLink
,以便使用AJAX将其添加到购物车中。这拒绝工作,如果我的ProductBean
是在RequestScope
,当我把它放在SessionScope
它工作,但只会添加产品1次。
据我所知,这应该是一个直接的RequestScope
,我不明白为什么它与SessionScope
一起工作。
我已经通读了这篇文章,但我不认为我违反了这些规则。
为了完整,这是我的ProductBean
:
import be.kdg.shop.model.stock.Product;
import java.util.logging.Logger;
import javax.enterprise.context.RequestScoped;
import javax.inject.Named;
@Named
@RequestScoped
public class ProductBean {
private static final Logger logger = Logger.getLogger(ProductBean.class.getName());
private Product product;
public ProductBean() {}
public Product getProduct() {
return product;
}
public void setProduct(Product product) {
this.product = product;
}
}
您的bean是请求作用域。因此,bean实例的寿命与单个HTTP请求-响应周期一样长。
当第一次请求带有表单的页面时,将创建一个新的bean实例,该实例接收一个具体的product
属性作为视图参数。在生成并发送相关响应之后,bean实例被废弃,因为它是请求的结尾。
当提交表单时,有效地触发一个新的HTTP请求,因此创建一个新的bean实例,将所有属性设置为默认值,包括product
属性。这样,对于整个请求,#{productBean.product}
就是null
。命令链接的父组件的rendered
属性将计算false
。因此,命令链接操作永远不会被解码。这与commandButton/commandLink/ajax action/listener方法未调用或输入值未更新的第5点相匹配,您已经发现,但显然没有真正理解。
@ViewScoped
。当您使用CDI而不是JSF来管理bean时,最好的选择是CDI @ConversationScoped
。这是相对笨拙的(你必须自己开始和结束作用域),所以一些CDI扩展,如MyFaces CODI,它提供了一个@ViewAccessScoped
可能更有用。
参见:
- 如何选择合适的bean scope?