女友不是想抛就抛,但异常却能够,不信请往下看。html
今天在MVC Controller中写代码时,纠结了一下:ajax
public async Task<ActionResult> Save(int? postId) { if(!IsOwner(postId.Value, userId)) { //抛不抛异常呢? } }
在这个地方要不要抛异常呢?json
若是不抛异常,就得这么写:app
public async Task<ActionResult> Save(int? postId) { if(!IsOwner(postId.Value, userId)) { return Json(new { isSuccess = false, message = "尝试执行未经受权的操做" }); } }
并且一般在这样的状况下,还须要记录日志,因而代码变成:async
if(!IsOwner(postId.Value, userId)) { Logger.Default.Info("UnauthorizedSave", "..."); return Json(new { isSuccess = false, message = "尝试执行未经受权的操做" }); }
若是抛异常呢,代码就能够这么写:post
if(IsOwner(postId.Value, userId)) { throw new UnauthorizedAccessException(); }
代码显得更简洁,更具表达力,并且记录错误日志能够在Application_Error中统一处理:url
protected void Application_Error(Object sender, EventArgs e) { var lastError = Server.GetLastError(); if (lastError != null) { Logger.Default.Error("Application_Error", lastError); Response.StatusCode = 500; Server.ClearError(); } }
但这会带来一个问题,客户端收到的将是自定义500错误页面的html代码,不只没有具体的错误信息,并且在ajax回调时还要额外处理,而一般咱们最期待的是一个json格式的返回结果。spa
因而要想实现“想抛就抛”,就必须解决:如何在Application_Error中统一处理ajax请求处理过程当中产生的异常,并将之转换成json格式的响应内容返回给客户端。日志
分解一下,就变成了2个问题:code
1)如何在Application_Error中判断一个请求是不是ajax请求?
今天上午经过MVC的扩展方法IsAjaxRequest解决了,详见如何在Global.asax中判断是不是ajax请求:
if ((new HttpRequestWrapper(Request)).IsAjaxRequest()) { }
2)如何生成json格式的响应内容并返回给客户端?
这个能够经过Json.NET+Response.Write来解决,代码以下:
Response.Clear(); Response.ContentType = "application/json; charset=utf-8"; Response.Write(Newtonsoft.Json.JsonConvert.SerializeObject( new { isSuccess = false, message = lastError.Message })); Response.Flush();
最终实现“想抛就抛”的代码以下:
protected void Application_Error(Object sender, EventArgs e) { var lastError = Server.GetLastError(); if (lastError != null) { CNBlogs.Infrastructure.Logging.Logger.Default.Error("Application_Error", lastError); if(Request != null && (new HttpRequestWrapper(Request)).IsAjaxRequest()) { Response.Clear(); Response.ContentType = "application/json; charset=utf-8"; Response.Write(Newtonsoft.Json.JsonConvert.SerializeObject( new { isSuccess = false, message = lastError.Message })); Response.Flush(); Server.ClearError(); return; } Response.StatusCode = 500; Server.ClearError(); } }
ajax客户端收到的结果以下:
今后,能够尽情地抛异常了。