小编典典
缺省情况下,如果在JSF
ajax请求期间发生异常,则无论操作是否成功执行,最终用户都不会获得任何形式的反馈。在Mojarra中,仅当项目阶段设置为“开发”时,最终用户才会看到仅包含异常类型和消息的裸JavaScript警报。
技术原因是默认情况下异步请求(读取:Ajax请求)不会返回同步响应(读取:整页)。相反,它们返回一些小指令和部分内容,说明如何更新已打开页面的HTML
DOM树。当发生异常时,基本上完全不存在这些指令。而是,一些错误信息被发送回。通常,您可以onerror在Ajax组件的属性中处理它们,例如显示警报或执行window.location更改。至少,这就是JSF对您的期望。
为了捕获和记录异常并有选择地更改整个响应,您基本上需要创建一个custom
ExceptionHandler。不幸的是,标准JSF没有提供默认的默认值(至少不是明智的选择)。在您的自定义异常处理程序中,您将可以使用Exception引起所有麻烦的实例。
这是一个启动示例:
public class YourExceptionHandler extends ExceptionHandlerWrapper {
private ExceptionHandler wrapped;
public YourExceptionHandler(ExceptionHandler wrapped) {
this.wrapped = wrapped;
}
@Override
public void handle() throws FacesException {
FacesContext facesContext = FacesContext.getCurrentInstance();
for (Iterator iter = getUnhandledExceptionQueuedEvents().iterator(); iter.hasNext();) {
Throwable exception = iter.next().getContext().getException(); // There it is!
// Now do your thing with it. This example implementation merely prints the stack trace.
exception.printStackTrace();
// You could redirect to an error page (bad practice).
// Or you could render a full error page (as OmniFaces does).
// Or you could show a FATAL faces message.
// Or you could trigger an oncomplete script.
// etc..
}
getWrapped().handle();
}
@Override
public ExceptionHandler getWrapped() {
return wrapped;
}
}
public class YourExceptionHandlerFactory extends ExceptionHandlerFactory {
private ExceptionHandlerFactory parent;
public YourExceptionHandlerFactory(ExceptionHandlerFactory parent) {
this.parent = parent;
}
@Override
public ExceptionHandler getExceptionHandler() {
return new YourExceptionHandler(parent.getExceptionHandler());
}
}
其中需要注册faces-config.xml如下:
com.example.YourExceptionHandlerFactory
或者,您可以继续使用OmniFaces。使用中的
page>配置,它将完全透明地确保异步请求期间的异常行为与同步请求期间的异常行为相同web.xml。
2020-07-26