我正在Glassfish v3上开发一个JSF2.0应用程序,我正在尝试处理ViewExpiredException。但是无论我做什么,我总是得到一个玻璃鱼错误报告,而不是我自己的错误页面。
为了模拟VEE的出现,我将以下函数插入到后台bean中,这将触发VEE。我正在通过一个commandLink从JSF页面触发这个函数。“守则”:
@Named
public class PersonHome {
(...)
public void throwVEE() {
throw new ViewExpiredException();
}
}一开始,我尝试将一个错误页面添加到我的web.xml中:
<error-page>
<exception-type>javax.faces.application.ViewExpiredException</exception-type>
<location>/error.xhtml</location>
</error-page> 但是这不起作用,我没有被重定向到错误,但是我看到了Glassfish错误页面,它显示了一个HTTP状态500页,包含以下内容:
description:The server encountered an internal error () that prevented it from fulfilling this request.
exception: javax.servlet.ServletException: javax.faces.application.ViewExpiredException
root cause: javax.faces.el.EvaluationException:javax.faces.application.ViewExpiredException
root cause:javax.faces.application.ViewExpiredException接下来,我尝试编写ExceptionHandlerFactory和CustomExceptionHandler,如JavaServerFaces 2.0中所描述的--完整的参考。因此,我将以下标记插入faces-config.xml中:
<factory>
<exception-handler-factory>
exceptions.ExceptionHandlerFactory
</exception-handler-factory>
</factory>并添加了以下类别:工厂:
package exceptions;
import javax.faces.context.ExceptionHandler;
public class ExceptionHandlerFactory extends javax.faces.context.ExceptionHandlerFactory {
private javax.faces.context.ExceptionHandlerFactory parent;
public ExceptionHandlerFactory(javax.faces.context.ExceptionHandlerFactory parent) {
this.parent = parent;
}
@Override
public ExceptionHandler getExceptionHandler() {
ExceptionHandler result = parent.getExceptionHandler();
result = new CustomExceptionHandler(result);
return result;
}
}自定义异常处理程序:
package exceptions;
import java.util.Iterator;
import javax.faces.FacesException;
import javax.faces.application.NavigationHandler;
import javax.faces.application.ViewExpiredException;
import javax.faces.context.ExceptionHandler;
import javax.faces.context.ExceptionHandlerWrapper;
import javax.faces.context.FacesContext;
import javax.faces.event.ExceptionQueuedEvent;
import javax.faces.event.ExceptionQueuedEventContext;
class CustomExceptionHandler extends ExceptionHandlerWrapper {
private ExceptionHandler parent;
public CustomExceptionHandler(ExceptionHandler parent) {
this.parent = parent;
}
@Override
public ExceptionHandler getWrapped() {
return this.parent;
}
@Override
public void handle() throws FacesException {
for (Iterator<ExceptionQueuedEvent> i = getUnhandledExceptionQueuedEvents().iterator(); i.hasNext();) {
ExceptionQueuedEvent event = i.next();
System.out.println("Iterating over ExceptionQueuedEvents. Current:" + event.toString());
ExceptionQueuedEventContext context = (ExceptionQueuedEventContext) event.getSource();
Throwable t = context.getException();
if (t instanceof ViewExpiredException) {
ViewExpiredException vee = (ViewExpiredException) t;
FacesContext fc = FacesContext.getCurrentInstance();
NavigationHandler nav =
fc.getApplication().getNavigationHandler();
try {
// Push some useful stuff to the flash scope for
// use in the page
fc.getExternalContext().getFlash().put("expiredViewId", vee.getViewId());
nav.handleNavigation(fc, null, "/login?faces-redirect=true");
fc.renderResponse();
} finally {
i.remove();
}
}
}
// At this point, the queue will not contain any ViewExpiredEvents.
// Therefore, let the parent handle them.
getWrapped().handle();
}
}但是,我仍然没有重定向到我的错误页面--我得到的HTTP 500错误和上面一样。我做错了什么,在我的实现中,没有正确地处理异常会缺少什么呢?任何帮助都非常感谢!
编辑
好吧,我很诚实。事实上,我的代码实际上是用Scala编写的,但这说来话长。我一直认为这是一个Java问题。这件事的真正错误是我自己的愚蠢。在我的(Scala)代码中,在CustomExceptionHandler中,我忘记了用“i.remove()”添加行;因此,ViewExpiredException在处理它之后一直驻留在UnhandledExceptionsQueue中,并且“冒泡”了。当它鼓起时,它就变成了一个ServletException。
我真的很抱歉把你们两个搞糊涂了!
发布于 2010-06-05 22:29:55
这个测试用例是假的。ViewExpiredException通常只在还原视图期间抛出(因为它在会话中丢失),而不是在呈现响应时抛出,也不实例化bean。在您的示例中,此异常是在实例化bean时抛出的,此异常被包装在ServletException中。
只有在HTTP会话过期时向服务器发送HTTP请求时,才会引发真正的ViewExpiredException。因此,基本上有两种方法可以可靠地再现这种情况:
h:form已经发布),关闭服务器并清理其工作目录(重要的是,因为大多数服务器将在关闭时将打开的会话序列化到磁盘,并在启动时取消序列化),重新启动服务器并提交已经打开的表单。将抛出一个ViewExpiredException。<session-timeout>在web.xml中设置为1分钟,并在打开带有POST表单的JSF页面后1分钟以上提交表单。这也会抛出ViewExpiredException。发布于 2010-06-05 14:51:50
我不是专家。这些只是胡乱的猜测或建议。
1)尝试重定向到标准的HTML页面,看看它是否有效,它应该与第一种方法本身一起工作,尝试编写一个基于此 PhaseListener,现在在恢复视图Phase.Right中抛出相同的异常,您正在调用应用程序或更新模型阶段。3)通过sysout,确保使用Servlet上下文配置了错误页(我没有尝试过,但应该是可能的)。
就连我都很好奇到底是什么问题!
https://stackoverflow.com/questions/2980708
复制相似问题