.net中自定義錯誤頁面的實(shí)現(xiàn)升級篇
問題描述:
在上一篇博文 “.net自定義錯誤頁面實(shí)現(xiàn)” 中已經(jīng)介紹了在.net中如何實(shí)現(xiàn)自定義錯誤頁面實(shí)現(xiàn)(有需要者可以去上一篇博文了解),單純按照上一篇博文那樣設(shè)置,能夠?qū)崿F(xiàn)所有請求的異常自定義跳轉(zhuǎn),但是這樣又會產(chǎn)生一個問題:當(dāng)通過ajax提交請求獲取接口提交請求,如果出現(xiàn)未處理的異常也會被重定向到自定義錯誤頁面。
針對ajax請求或者接口請求,這樣返回一個重定向頁面,用戶體驗(yàn)顯然不是太友好,針對這個問題,下面簡單總結(jié)一下我自己的想法和解決方案,當(dāng)然不一定科學(xué)和合理,所以也希望有大牛多多指點(diǎn)。
解決思路,我想到的有二:
解決方案一:
從物理結(jié)構(gòu)上分割,將web項(xiàng)目嚴(yán)格分割成兩個項(xiàng)目(當(dāng)然可根據(jù)需要繼續(xù)細(xì)分):網(wǎng)站(只有網(wǎng)站頁面資源等內(nèi)容)、接口(包括網(wǎng)站的所有數(shù)據(jù)邏輯處理,頁面的數(shù)據(jù)請求交互都是直接同接口交互(js技術(shù))),只是網(wǎng)站項(xiàng)目按照上一篇博文方式設(shè)置自定義錯誤頁面方式,這樣是能夠解決問題,項(xiàng)目也會更加的清晰,也有很多公司的項(xiàng)目就是按照這種方式(尤其是webApp),但是在實(shí)際項(xiàng)目中,很多項(xiàng)目是沒有達(dá)到這種嚴(yán)格區(qū)分的,所以下面的解決方案二,將介紹一個更通用的方式
解決方法二:
解決思路是:將上一篇博文.net自定義錯誤頁面實(shí)現(xiàn) 與 上上一篇博文 .net捕捉全局未處理異常的3種方式 結(jié)合使用,并在實(shí)際開發(fā)中嚴(yán)格約定(出了url地址請求以外的其他請求都通過post請求實(shí)現(xiàn)交互),在撲捉到異常時,如果是post請求,處理異常,并清除異常。具體以步驟如下:
第一步:定義一個請求處理結(jié)果數(shù)據(jù)MODEL,代碼如下:
/// <summary> /// 請求結(jié)果MRequestResult /// </summary> public class MRequestResult { /// <summary> /// 請求結(jié)果編碼(是一個枚舉值) /// </summary> private RequestResultCodeEnum requestResultCode; /// <summary> /// 處理結(jié)果具體的返回值 /// </summary> private object resultValue; /// <summary> /// 請求結(jié)果編碼(是一個枚舉值) /// </summary> public RequestResultCodeEnum RequestResultCode { get { return this.requestResultCode; } set { this.requestResultCode = value; } } /// <summary> /// 處理結(jié)果具體的返回值 /// </summary> public object ResultValue { get { return this.resultValue; } set { this.resultValue = value; } } } /// <summary> /// 請求結(jié)果編碼枚舉值() /// </summary> public enum RequestResultCodeEnum { /// <summary> /// 請求成功 /// </summary> Success = 1, /// <summary> /// 請求失敗 /// </summary> Fail = -1, }
第二步:按照 上一篇博文: .net自定義錯誤頁面實(shí)現(xiàn)的步驟,配置好自定義錯誤頁面相關(guān)配置操作
第三步:按照 上上一篇博文:.net捕捉全局未處理異常的3種方式 的步驟實(shí)現(xiàn)全局異常為處理相關(guān)操作設(shè)置
第四步:在撲捉全局未處理的異常中,添加上針對post請求的異常處理過濾(直接輸入封裝后的),具體代碼如下:
/// <summary> /// 異常處理 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> public void context_Error(object sender, EventArgs e) { //此處處理異常 HttpContext ctx = HttpContext.Current; HttpResponse response = ctx.Response; HttpRequest request = ctx.Request; //獲取到HttpUnhandledException異常,這個異常包含一個實(shí)際出現(xiàn)的異常 Exception ex = ctx.Server.GetLastError(); //實(shí)際發(fā)生的異常 Exception iex = ex.InnerException; //// 異常日志落地 //// 異常日志落地包括:記錄異常文本文件日志、或者記錄異常日志數(shù)據(jù)庫等等(根據(jù)實(shí)際項(xiàng)目需要做記錄) //// 獲取請求方法 string httpMethod = request.HttpMethod; //// 如果是POST請求,那么是以下請求之一 //// ajax接口請求 //// form表單提交 //// 這種情況的異常,不用跳轉(zhuǎn)至自已的異常錯誤頁面,直接返回對應(yīng)系統(tǒng)異常 if (httpMethod.ToUpper() == "POST") { //// 構(gòu)建返回對象值 MRequestResult requestResultM = new MRequestResult(); requestResultM.RequestResultCode = RequestResultCodeEnum.Fail; requestResultM.ResultValue = "系統(tǒng)異常,請聯(lián)系管理員!"; response.Write(JsonConvert.SerializeObject(requestResultM, Formatting.Indented)); ctx.Server.ClearError();//處理完及時清理異常 } }
代碼實(shí)例:
ajax請求方法及其邏輯處理實(shí)例代碼:
$(function () { $.ajax({ async: true, type: "post", url: "../ActionTestResult/ContentResultTest", data: "name=xu", success: function (resultValue) { if (resultValue) { //// 解析處理結(jié)果 var resultObj = $.parseJSON(resultValue); //// 當(dāng)RequestResultCode==1 說明該請求成功 ////(備注:并不代表處理成功,具體是否處理成功需要通過ResultValue的值根據(jù)接口約定解析做相應(yīng)的邏輯處理) if (resultObj["RequestResultCode"] == 1) { //// 自定義請求成功邏輯處理 //// 通過解析具體的ResultValue的值做相應(yīng)的邏輯處理..... if (resultObj["ResultValue"]) { var doResult = resultObj["ResultValue"].split('^'); if (doResult && doResult.length > 1) { if (doResult[0] == 1) { //// 說明處理成功,做相應(yīng)的邏輯處理 alert("處理成功!"); } else { //// 處理失敗 alert(doResult[1]); } } else { alert("操作失?。?); } } else { alert("未知結(jié)果"); } } else { //// 說明請求異常 //// 自定義邏輯處理 alert(resultObj["ResultValue"]); } } else { //// 自定義邏輯處理 alert("操作失??!"); } console.log(resultValue); }, error: function (data) { //// 自定義邏輯處理 alert("系統(tǒng)異常,請聯(lián)系管理員。電話:8888888"); console.log(data); } }); });
ajax對應(yīng)的后臺請求接受實(shí)例代碼:
/// <summary> /// 測試 /// </summary> /// <returns></returns> public ContentResult ContentResultTest(string name) { //// 構(gòu)建請求處理結(jié)果Model MRequestResult requestResultM = new MRequestResult() { RequestResultCode = RequestResultCodeEnum.Success }; //// 業(yè)務(wù)處理結(jié)果 string doResult = string.Empty; //// 本次自作簡單的參數(shù)非空判斷,只一個示例 //// 處理結(jié)果本例中也只是通過^鏈接表示,在實(shí)際處理過程中,可以將結(jié)果通過一個Json字符串 if (string.IsNullOrEmpty(name)) { doResult = "-1^操作失敗:名稱不能為空!"; } else { ///// 其他自定義業(yè)務(wù)邏輯處理,此處省略..... doResult = "1^操作成功"; } requestResultM.ResultValue = doResult; //// 返回結(jié)果 return Content(JsonConvert.SerializeObject(requestResultM, Formatting.Indented)); }
是不是覺得說的有點(diǎn)繞,本人表述能力有限,不懂的評論交流。
最后:個人能力有限也許該解決方式并不友好,有更好的解決方法,也歡迎留言交流,多多指點(diǎn),多多指教
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
CheckBox為CheckBoxList實(shí)現(xiàn)全選或全取消選擇(js代碼實(shí)現(xiàn))
在管理商品后臺是,由于CheckBoxList的選擇太多,用戶需要一個全選或全取消的功能,這樣操作起來會提高效率同時可以減少誤點(diǎn)等,本文將教大家如何實(shí)現(xiàn),有需要的朋友可以參考下,望本文對你有所幫助2013-01-01Asp.net回調(diào)技術(shù)Callback學(xué)習(xí)筆記
這篇文章主要記錄了Asp.net回調(diào)技術(shù)Callback的一些知識,感興趣的朋友可以參考下2014-08-08