我在应用程序启动页面加载时收到以下错误:
SEVERE: Error Rendering View[/HomeTemplate/equityVolume.xhtml]
javax.el.ELException: /HomeTemplate/equityVolume.xhtml @70,78 value="#{equityBean.scripList}": java.lang.IllegalStateException: PWC3999: Cannot create a session after the response has been committed...
Caused by: java.lang.IllegalStateException: PWC3999: Cannot create a session after the response has been committed...
当我将css应用到我的主页时出现此错误,当我删除css模板时错误消失(但我想要使用css模板) 以下是导致错误的bean代码片段(通过调试找到)
public List<MasterScrip> getScripList() {
HttpServletRequest req=(HttpServletRequest)FacesContext.getCurrentInstance().getExternalContext().getRequest(); //error line
HttpSession session=req.getSession();
type=(String)session.getAttribute("type");...
xhtml代码:
<h:body>
<ui:composition template="commonClientLayout.xhtml">
<ui:define name="contentFile">
<div id="content">
<h:form id="frm">...
当我删除ui:composition并定义标签(即如果我不应用css),那么我不会得到这个错误。 什么可能导致这个错误,我该如何解决?
编辑:
@PostConstruct
void initialiseSession() {
if(type!=null)
{
if(type.equalsIgnoreCase("losers"))
{
scripList=new ArrayList<MasterScrip> ();
scripList=getScripByPriceLosers(exchange);
// return scripList;
}
else if(type.equalsIgnoreCase("gainers"))
{
scripList=new ArrayList<MasterScrip> ();
scripList=getScripByPriceGainers(exchange);
// return scripList;
}
else
{
scripList=new ArrayList<MasterScrip> ();
scripList=getScripByVolumeType(exchange);
// return scripList;
}
}
else
{
scripList=new ArrayList<MasterScrip> ();
scripList=getScripByVolumeType(exchange);
}
}
public List<MasterScrip> getScripList() {
return scripList;
}
再次修改:
SEVERE: Error Rendering View[/equityVolume.xhtml]
java.lang.IllegalStateException
at org.apache.catalina.connector.ResponseFacade.setBufferSize(ResponseFacade.java:275)...
编辑:web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app version="3.0" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd">
<context-param>
<param-name>javax.faces.PROJECT_STAGE</param-name>
<param-value>Production</param-value>
</context-param>
<context-param>
<param-name>javax.faces.FACELETS_BUFFER_SIZE</param-name>
<param-value>65535</param-value>
</context-param>
<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>
<session-config>
<session-timeout>
30
</session-timeout>
</session-config>
<welcome-file-list>
<welcome-file>equityVolume.xhtml</welcome-file>
</welcome-file-list>
<security-constraint>
<display-name>Constraint1</display-name>
<web-resource-collection>
<web-resource-name>AdminTemplate</web-resource-name>
<description/>
<url-pattern>/AdminTemplate/*</url-pattern>
</web-resource-collection>
<auth-constraint>
<description/>
<role-name>admin</role-name>
</auth-constraint>
</security-constraint>
<security-constraint>
<display-name>Constraint2</display-name>
<web-resource-collection>
<web-resource-name>ClientTemplate</web-resource-name>
<description/>
<url-pattern>/ClientTemplate/*</url-pattern>
</web-resource-collection>
<auth-constraint>
<description/>
<role-name>client</role-name>
</auth-constraint>
</security-constraint>
<login-config>
<auth-method>FORM</auth-method>
<realm-name>DataRealm</realm-name>
<form-login-config>
<form-login-page>/equityVolume.xhtml</form-login-page>
<form-error-page>/errorpage.xhtml</form-error-page>
</form-login-config>
</login-config>
<security-role>
<description/>
<role-name>admin</role-name>
</security-role>
<security-role>
<description/>
<role-name>client</role-name>
</security-role>
</web-app>
答案 0 :(得分:6)
你不应该在吸气者做生意。而是在bean(post)构造函数中执行它。
您的具体问题是由于您在尚未创建服务器HttpSession
的全新浏览器会话上请求相对较大的页面而且相对引用EL表达式#{equityBean.scripList}
在页面后面。
响应缓冲区默认为2KB,当由于响应较大而溢出时,它将被提交。这意味着将发送所有响应头,并且将发送前2~KB的HTML输出。然后,在那之后,将解析EL表达式#{equityBean.scripList}
,其中您正在尝试获取会话。如果此时尚未创建服务器HttpSession
,则服务器需要在响应头中设置cookie,以便为后续请求维护它。但如果响应已经提交,那当然是不可能的。因此这个例外。
如上所述,只需在bean的(post)构造函数中完成工作。或者只是将其作为托管属性注入。
@ManagedProperty("#{type}")
private String type;
如果异常仍然存在,您可能正在使用较早版本的Mojarra,该版本受到问题2215和2277中所述的错误的影响,该问题也是由引起的极端推迟“不必要的”会话创建。自Mojarra 2.1.8以来,这已得到修复。因此,升级到它或更新(目前是2.1.9)应该这样做。
无关,getScripList()
逻辑就像臭臭一样。但这是一个不同的问题/问题的主题。您是否知道只能在EL中访问名为“type”的会话属性#{type}
?您是否意识到在JSF支持bean类中进行原始javax.servlet.*
导入通常表明您可能以错误的方式执行操作,并且可能有更多“JSF-ish”方法来实现具体的功能要求?
答案 1 :(得分:3)
我不知道这个Web框架(它是JSF吗?)但是这里发生了什么。您的XHTML开始呈现输出,并且已经将一些字符发送到浏览器。这意味着整个标题也被发送了。
在模板中间的某个地方,您正在调用#{equityBean.scripList}
(BTW拼写错误),然后调用:
HttpSession session=req.getSession();
如果不存在,则此方法将创建HTTP会话。必须将会话ID与响应(使用cookie或URL重写)一起发送回客户端,以便在后续请求中识别会话。但是,由于响应标头已经发送,因此servlet容器无法返回会话ID,并抛出异常,避免出现更大的问题。
你怎么解决这个问题?当绝对没有与请求关联的会话时,您似乎是第一次呈现页面。您可以避免创建会话:
HttpSession session=req.getSession(false); //false here!
if(session != null) {
type=(String)session.getAttribute("type");
}
//handle the case when session or type attribute weren't there
另一种方法是在将控件传递给视图之前主动创建会话。但是,您仍需要检查type
的{{1}}属性。