在提交响应后无法创建会话

时间:2012-06-08 13:13:29

标签: java session jsf java-ee managed-bean

我在应用程序启动页面加载时收到以下错误:

 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>

2 个答案:

答案 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,该版本受到问题22152277中所述的错误的影响,该问题也是由引起的极端推迟“不必要的”会话创建。自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}}属性。