从ADF Faces JSF 1.2中的托管bean构造函数导航

Tap*_*ose 4 navigation jsf exception-handling oracle-adf jsf-1.2

是否可以从托管bean的构造函数导航到另一个页面/视图?如果发生任何异常,我想要这个重定向.我尝试了很多方法:

尝试-1:

getFacesContext().responseComplete();
getFacesContext().getApplication().getNavigationHandler().handleNavigation(getFacesContext(), null, "gotoPartError");    
getFacesContext().renderResponse();
Run Code Online (Sandbox Code Playgroud)

尝试-2:

getServletResponse().sendRedirect("partError.jspx")
Run Code Online (Sandbox Code Playgroud)

尝试-3:

getFacesContext().responseComplete();    
getFacesContext().getExternalContext().redirect(getServletRequest().getContextPath() + "/pages/partError.jspx");
Run Code Online (Sandbox Code Playgroud)

尝试-4:

RequestDispatcher dispatcher = getServletRequest().getRequestDispatcher("partError.jspx");
dispatcher.forward(getServletRequest(), getServletResponse());
Run Code Online (Sandbox Code Playgroud)

尝试-5:

FacesContext context = getFacesContext();
UIViewRoot newPage = context.getApplication().getViewHandler().createView(context, "/partError.jspx");
context.setViewRoot(newPage);
context.renderResponse();
Run Code Online (Sandbox Code Playgroud)

尝试-6:

ControllerContext.getInstance().getCurrentViewPort().setViewId("partError");
Run Code Online (Sandbox Code Playgroud)

尝试-7:

Exception Handler in adfc-config.xml
Run Code Online (Sandbox Code Playgroud)

尝试-8:

Custom service handler defined in /.adf/META-INF/services/oracle.adf.view.rich.context.Exceptionhandler which extends oracle.adf.view.rich.context.Exceptionhandler
Run Code Online (Sandbox Code Playgroud)

尝试-9:

By extending JSF Life Cycle
Run Code Online (Sandbox Code Playgroud)

他们都没有工作.对于我收到的所有案件

java.lang.IllegalStateException: Cannot forward after response has been committed
Run Code Online (Sandbox Code Playgroud)

在JSF 1.2中真的不可能吗?当我使用ADF 11.1.1.6.0,它使用JSF 1.2时,上面的一些"Try"包含ADF Faces方法.

无论如何,我需要,可以是JSF 1.2或ADF Faces,导航到错误页面.我获得成功的唯一方法是使用从后端执行的javascript,以便在出现错误时在_self窗口中打开错误页面,但我不是很喜欢它.

这件事中的任何指针都会非常有用.

Bal*_*usC 5

如果理解问题的原因,则更容易解决问题.一个很好的例外基本上已经说明了问题原因的一切.

仔细看看:

java.lang.IllegalStateException:在提交响应后无法转发

响应已经提交.这是一个不归路.也许您无法理解提交响应意味着什么(其结果是您也无法理解异常本身).

默认情况下,HTTP响应被写入缓冲区,每隔~2KB刷新一次,具体取决于服务器配置.刷新响应缓冲区会导致写入的字节实际从服务器发送到客户端.一旦第一次发生这种情况,就会认为响应已提交.这是一个不归路.如果服务器实际上需要在之后更改响应,则服务器无法从客户端返回已写入的字节.

如果您有一些可能需要更改响应的代码,那么您应该在提交响应之前调用它.

在您的特定情况下,在生成HTML输出期间,托管bean显然是在JSF呈现响应阶段中构建的.已生成的HTML输出的一部分已发送到客户端(因此,响应已提交).你显然引用请求JSF页面范围的bean相对较晚,或响应缓冲区比较小,或者HTML <head>是一种使得已经刷新了之前比较大的<body>启动,等等.

您确实需要在呈现响应阶段之前调用代码.在JSF 1.2中,您可以使用<f:view beforePhase>for.

例如

<f:view beforePhase="#{bean.navigate}">
Run Code Online (Sandbox Code Playgroud)

public void navigate(PhaseEvent event) {
    if (event.getPhaseId() == PhaseId.RENDER_RESPONSE) {
        // Do here your job which should run right before the RENDER_RESPONSE.
    }
}
Run Code Online (Sandbox Code Playgroud)

然后你的Try-1和Try-3会起作用(你可以将这些responseComplete()renderResponse()线条留下,它们已被隐含地处理掉).

Try-2和Try-4很差.你应该避免javax.servlet.*在你的支持bean 中导入.Try-5很笨拙.Try-6,Try-7和Try-8超出了我的范围.Try-9是可行的,但非常笨拙.