【ASP.NET Core】處理異常(上篇)

来源:https://www.cnblogs.com/tcjiaan/archive/2018/02/23/8461408.html
-Advertisement-
Play Games

依照老周的良好作風,開始之前先說點題外話。 前面的博文中,老周介紹過自定義 MVC 視圖的搜索路徑,即向 ViewLocationFormats 列表添加相應的內容,其實,對 Razor Page 模型,也可以向 PageViewLocationFormats 列表添加相應的搜索路徑,比如 /MyP ...


依照老周的良好作風,開始之前先說點題外話。

前面的博文中,老周介紹過自定義 MVC 視圖的搜索路徑,即向 ViewLocationFormats 列表添加相應的內容,其實,對 Razor Page 模型,也可以向 PageViewLocationFormats 列表添加相應的搜索路徑,比如 /MyPages/{1}/{0}.cshtml。其中,0 是視圖名,1 是頁面名稱。比如這樣。

            services.AddMvc().AddRazorOptions(opt =>
            {
                opt.ViewLocationFormats ...
                opt.PageViewLocationFormats ...
            });

然而,我們知道,基於 Razor 的 Web Page 模型是以頁面為單位的,也就是說路徑路由是直接指向頁面的(不包含.cshtml 擴展名),即不需要 MVC 模型的路由方式。所以,我們並不需要修改 PageViewLocationFormats 中的內容。許多時候,我們只要告訴應用程式在哪個目錄下查找 Page 就行了。

預設的搜索位置是 /Pages 目錄,我們可以通過以下代碼來修改。

        public void ConfigureServices(IServiceCollection services)
        {
            services.AddMvc().AddRazorPagesOptions(opt =>
            {
                opt.RootDirectory = "/UI";
            });
        }

以上代碼寫在 Startup 類中,這個應該明白吧。RootDirectory 就是用來指定應用程式查找 Razor 頁面的根目錄路徑,此處我指寫了 /UI,所以,在我的項目中,我只要建一個 UI 目錄,然後各類 Razor 頁就往裡面放就行了。

 

 ========================================================================================

好了,題外話扯完了,開始說正題吧。今天咱們聊聊有關異常處理的破事吧,也可以說是錯誤處理,反正就這個意思,你理解就好,專業名詞不必較勁,只有那些吃飽了撐著的“學術人才”才會跟名詞較勁。

老辦法,咱們結合示例來講述,這樣各位觀眾不會乏味。

大家知道,娛樂產品腎Phone已經成為流行玩具,近年來,購買腎Phone不一定只能用貨幣,比較典型的一種支付方式是賣腎買Phone。說實話,現在許多國產娛樂產品也很便宜,配置也不錯,幾百塊錢就能玩得刷刷響了,割腎真沒什麼必要。

為了方便人們以腎換 Phone ,老周特意開發了一個線上賣腎系統。大致流程是這樣的,如果你有閑置的腎,可以打開主頁,輸入你的一些信息,然後報個價,其他用戶看見後,如果覺得合理,就認購此腎。

 

 為了使操作流程更簡單,易上手,輕入門,該平臺只需要輸入姓名和腎的價格即可參加報價。

 

大致的頁面代碼如下。

        <form method="post">
            <div class="form-group">
                <label for="name">姓名:</label>
                <input type="text" class="form-control" name="name"/>
            </div>
            <div class="form-group">
                <label for="price">價格:</label>
                <input type="number" name="price" class="form-control"/>
            </div>
            <div class="form-group">
                <button type="submit" class="btn btn-success w-100">提  交</button>
            </div>
        </form>

Razor 頁面很像我們以前玩過的 aspx 頁面,每個頁面都配套一個隱藏代碼文件。Razor 頁也會配有一個頁面模型類,註意這個模型類要從 PageModel 派生,不是 Page 類,別搞錯了,Page 類只是作為生成 HTML 代碼的基類,我們的 .cshtml 文件在預編譯後,是隱式繼承自 RazorPage 類的。除非你要開發自己的標記語言,否則你不必理會這些類。

記住了,與 Razor 頁關聯的模型類是從 PageModel 類派生的,比如,本例中,當有人填寫了閑置腎的相關信息後,以 POST 方式提交,這是候,如果頁面模型類中包含了名字為 OnPost、OnPostAsync ……的方法時,就會自動調用。如果想把我們上面那個 form 中的 name 和 price 的值傳遞給方法,直接讓 OnPost 方法的參數與 form 中的元素名稱相同就可以了。

    public class IndexModel : PageModel
    {
       public IActionResult OnPost(string name, decimal price)
        {
            if (string.IsNullOrWhiteSpace(name))
            {
                throw new Exception("你怎麼不留下姓名啊,賣腎又不是丟人的事。");
            }
            if(price <= 0.0M)
            {
                throw new Exception("靠!你的腎這麼不值錢嗎?還免費送,包郵不?");
            }
            return RedirectToPage("/Success");
        }
    }

OnPost 不是 PageModel 基類的方法,而是我們自己寫的,只是代碼約定,Asp.net Core 裡面用到很多代碼約定,它在運行的時候會查找這些特定的名字。

上面代碼中,還對傳遞進來的 form 值進行驗證,如果不符合要求,會拋出異常。

 

一般來說,在 Startup 類的 Configure 方法中,我們會判斷一下,如果應用程式處於開發階段,為了方便測試,應該加入這些代碼。

      if (env.IsDevelopment())
      {
           app.UseDeveloperExceptionPage();
      }

這樣,我們在測試時能看到詳細的異常信息。

 

但是,在實際便用時,我們不能公開這麼詳細的信息,這樣容易勾起人們的犯罪衝動。所以,一般會添加一個頁面,專門用來顯示錯誤信息。比如:

@page

<div class="card">
    <div class="card-header bg-danger">
        <span class="text-light">錯誤</span>
    </div>
    <div class="card-body">
        <span class="card-text">唉,真抱歉。你提交的腎不符合國際標準,沒人要的。</span>
    </div>
</div>

 

然後我們要在 Startup.Configure 方法中配置一下。

  app.UseExceptionHandler("/Error");

加上這一行後,當發生異常時,就會跳轉到 /Error 頁面。

 

 不過,你也許會覺得,雖然不能公開異常信息,但一些必要的描述應該要的,不然,用戶不知道發生了啥事。我們可以通過 HttpContext 的 Features 集合獲取一個用來處理異常的 Feature,它的原型介面是 IExceptionHandlerFeature,我們不必關心它的實現類型是誰,只要訪問它的 Error 屬性就能得到關聯的 Exception 實例。

因此,我們的錯誤頁可以改一下。

@page
@using Microsoft.AspNetCore.Diagnostics
@{
    IExceptionHandlerFeature exf = HttpContext.Features.Get<IExceptionHandlerFeature>();
    Exception ex = exf?.Error;
}

<div class="card">
    <div class="card-header bg-danger">
        <span class="text-light">錯誤</span>
    </div>
    <div class="card-body">
        @if (ex == null)
        {
            <span class="card-text">唉,真抱歉。你提交的腎不符合國際標準,沒人要的。</span>
        }
        else
        {
            <span class="card-text">@ex.Message</span>
        }
    </div>
</div>

通過以下代碼獲得異常實例的引用。

    IExceptionHandlerFeature exf = HttpContext.Features.Get<IExceptionHandlerFeature>();
    Exception ex = exf?.Error;

這樣就可以在頁面上顯示異常的描述信息了。

 

 

 可能你又想到了,我不想輸出個頁面,我只想返回一些簡單的文本,那麼,你在 Startup.Configure 中可以這樣寫。

            app.UseExceptionHandler(x =>
            {
                x.Run(async context =>
                {
                    var ex = context.Features.Get<Microsoft.AspNetCore.Diagnostics.IExceptionHandlerFeature>()?.Error;
                    string msg = ex == null ? "發生錯誤。" : ex.Message;
                    context.Response.ContentType = "text/plain;charset=utf-8";
                    await context.Response.WriteAsync(msg);
                });
            });

裡面的變數 x 就是當前的 IApplicationBuilder ,與傳遞給 Configure 方法的 app 參數類型一樣,這時候我們可以用 Reponse 的方法返回自定義的文本。

 

 好了,今天的內容就介紹到這兒吧,其實異常處理還有一種方法——使用 Filter,這個咱們留到下一篇博文再和大伙分享。

本文示例源代碼下載

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 題目背景 無 題目描述 有兩堆石子,數量任意,可以不同。游戲開始由兩個人輪流取石子。游戲規定,每次有兩種不同的取法,一是可以在任意的一堆中取走任意多的石子;二是可以在兩堆中同時取走相同數量的石子。最後把石子全部取完者為勝者。現在給出初始的兩堆石子的數目,你先取,假設雙方都採取最好的策略,問最後你是勝 ...
  • 對資料庫操作的常用方法 ...
  • 引言 隨著手機及移動設備的普及,移動端的應用也進入了熱潮。以前PC端的門戶網站,大多也均推出了適配移動設備的網站或者APP,再差的也註冊了個公眾號。在移動應用開發中,目前據我所瞭解到的解決方案有:1、原生態APP開發;2、適配移動設備的JS+HTML網站,套上跨平臺的“馬甲”;3、微信小程式及公眾號 ...
  • var array=['REG','2018','2018']; array.indexOf(‘REG’) // 0 array.indexOf(‘R’) // -1 array.indexOf(’2018′) // 1 array.indexOf(2018) // -1 var array=['R ...
  • Routing 到目前為止,我們已經解決了MVC的很多問題,但忽略了最基本的最重要的一個問題:當用戶發送請求時,會發生什麼? 最好的答案是“執行Action 方法”,但仍存在疑問:對於一個特定的URL請求,如何確定控制器和action 方法。在開始實驗31之前,我們首先來解答上述問題,你可能會困惑為 ...
  • 在前面的文章中,我們已經知道如何合併、拆分多個PDF文件,在這篇文章中的合併、拆分PDF文檔主要是以方便文檔管理的目的來操作文檔,在文檔查閱、管理及存儲上很方便實用。但是我們如果想要合併多個文檔中的部分文檔頁的內容,該如何來做呢?可以參考接下來將要介紹的合併方法。 PS: 本篇文章是對Free Sp ...
  • using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.... ...
  • IL DASM反編譯工具 使用C#的猿人或多或少都會對微軟的IL反編譯工具(ildasm.exe)有所認識。我最早接觸到這工具是公司同事使用他反編譯exe程式,進行研讀和修改。感覺他還是很強大。 IL是微軟平臺上的一門中間語言,我們常寫的C#代碼在編譯器中都會自動轉換成IL,然後在由即時編譯器(JI ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...