我在页面上为不同的表单多次创建@ViewScoped
个bean时出现问题。
我创建了一个简单的测试页面:
<?xml version="1.0" encoding="UTF-8" ?>
<!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:f="http://java.sun.com/jsf/core"
xmlns:h="http://java.sun.com/jsf/html">
<h:head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
</h:head>
<h:body>
<h:form id="f1">
<h:commandButton value="action 1" action="#{testController.action}">
<f:ajax />
</h:commandButton>
</h:form>
<h:form id="f2">
<h:commandButton value="action 2" action="#{testController.action}">
<f:ajax />
</h:commandButton>
</h:form>
</h:body>
</html>
和一个支持bean:
@ManagedBean
@ViewScoped
public class TestController implements Serializable {
private static final long serialVersionUID = 1L;
public TestController() {
System.out.println("TestController created");
}
public String action() {
System.out.println("action() invoked");
return null;
}
private void writeObject(ObjectOutputStream out) throws IOException {
System.out.println("serialization");
out.defaultWriteObject();
}
private void readObject(ObjectInputStream in) throws IOException,
ClassNotFoundException {
System.out.println("deserialization");
in.defaultReadObject();
}
}
这是web.xml:
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
id="WebApp_ID" version="3.0">
<display-name>WebTest</display-name>
<welcome-file-list>
<welcome-file>index.xhtml</welcome-file>
</welcome-file-list>
<servlet>
<servlet-name>Faces Servlet</servlet-name>
<servlet-class>javax.faces.webapp.FacesServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>Faces Servlet</servlet-name>
<url-pattern>*.xhtml</url-pattern>
</servlet-mapping>
<context-param>
<param-name>javax.faces.FACELETS_REFRESH_PERIOD</param-name>
<param-value>1</param-value>
</context-param>
<context-param>
<param-name>javax.servlet.jsp.jstl.fmt.localizationContext</param-name>
<param-value>resources.application</param-value>
</context-param>
<context-param>
<param-name>javax.faces.STATE_SAVING_METHOD</param-name>
<param-value>client</param-value>
</context-param>
<listener>
<listener-class>com.sun.faces.config.ConfigureListener</listener-class>
</listener>
</web-app>
当我单击action 1
按钮时,会创建一个新的TestController
bean,然后使用writeObject
方法序列化。然后,当我单击action 2
按钮时,我希望它将被反序列化,而是创建一个新bean,并且这两个bean对每个<h:commandButton>
独立行动。
我在Mojarra和Myfaces都遇到过这个问题,所以我猜这不是图书馆里的错误。
也许我做了一些愚蠢的错误或错过了一些东西,但是不应该@ViewScoped
bean是当前视图的唯一bean,而不管页面上<h:form>
的数量是多少?
更新:我发现删除<f:ajax>
代码后问题消失了;不幸的是,这不是我的选择。
更新2:我认为我找到了问题的原因:使用多个表单时,只会自动更新请求表单ViewState
。在我的示例中,如果我使用第一个表单发送请求,则第二个表单的视图状态不会更新,除非我在第一个表单中将render=":f2"
添加到<f:ajax>
。并且由于@ViewScoped
bean存储在视图状态中,因此第二种形式的视图状态将不包含此bean,因此在从第二种形式发送请求时将创建新的bean。因此,如果我的推理成立,我必须使用render
属性手动更新视图中的所有表单,以使表单中的视图状态保持一致。但这似乎很不方便,特别是如果页面上有很多表单。没有其他解决方案吗?
答案 0 :(得分:0)
到目前为止发布的代码并没有这样做。至少,不适用于默认的JSF游乐场项目。
然而,当您将视图范围的bean属性绑定到标记处理程序的任何属性(如JSTL <c:forEach>
,Facelets <ui:include>
,JSF <f:attribute>
或{的时候,症状是可识别的。 JSF UI组件的{1}}或id
属性。
在构建和恢复视图期间,将执行标记处理程序属性中的任何EL表达式以及UI组件的binding
和id
属性。在回发期间恢复视图时,视图范围bean仅在恢复视图后可用(因为它们存储在那里)。因此,在视图恢复的时刻它们不可用,这就解释了为什么它们每次都被构建。