asp.net core系列 38 WebAPI 返回類型與響應格式--必備

来源:https://www.cnblogs.com/MrHSR/archive/2019/03/06/10477603.html
-Advertisement-
Play Games

一.返回類型 ASP.NET Core 提供以下 Web API Action方法返回類型選項,以及說明每種返回類型的最佳適用情況: (1) 固定類型 (2) IActionResult (3) ActionResult<T> 1.1 固定類型 最簡單的操作是返回基元或複雜數據類型(如 string ...


一.返回類型

  ASP.NET Core 提供以下 Web API Action方法返回類型選項,以及說明每種返回類型的最佳適用情況:

  (1) 固定類型

  (2) IActionResult

  (3) ActionResult<T>

  

  1.1 固定類型

    最簡單的操作是返回基元或複雜數據類型(如 string 或自定義對象類型)。 請參考以下Action,該Action返回自定義 Product 對象的集合:

[HttpGet]
public IEnumerable<Product> Get()
{
    return _repository.GetProducts();
}

     適用場景:在執行Action期間,無需要根據條件判斷返回不同類型,只返回固定類型即可滿足要求。 上述操作不接受任何參數,因此不需要參數約束驗證。

 

  1.2  IActionResult類型

    Action方法中可能有多個 ActionResult 返回類型時,適合使用 IActionResult 返回類型。ActionResult 類型可以表示多種 HTTP 狀態代碼。 屬於此類別的一些常見返回類型包括:BadRequestResult (400)、NotFoundResult (404) 和 OkObjectResult (200)。

    由於Action方法中有多個返回類型和路徑,因此必須使用 [ProducesResponseType] 特性。 此特性可針對 Swagger 等工具生成的 API 幫助頁生成更多描述性響應詳細信息(上篇有介紹)。 [ProducesResponseType] 指示Action將返回的已知類型和 HTTP 狀態代碼。

    下麵是一個同步Action,該Action方法中可能有兩種返回類型:

[HttpGet("{id}")]
[ProducesResponseType(typeof(Product), StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
public IActionResult GetById(int id)
{
    if (!_repository.TryGetProduct(id, out var product))
    {
        return NotFound();
    }

    return Ok(product);
}

     下麵是一個非同步Action,該Action方法中可能有兩種返回類型:

[HttpPost]
[ProducesResponseType(typeof(Product), StatusCodes.Status201Created)]
[ProducesResponseType(StatusCodes.Status400BadRequest)]
public async Task<IActionResult> CreateAsync([FromBody] Product product)
{
    if (product.Description.Contains("XYZ Widget"))
    {
        return BadRequest();
    }

    await _repository.AddProductAsync(product);

    return CreatedAtAction(nameof(GetById), new { id = product.Id }, product);
}

     適用場景:當Action方法中可能有多個 ActionResult 返回類型時,適合使用 IActionResult 返回類型。

 

  1.3 ActionResult<T>

     ASP.NET Core 2.1 引入了 ActionResult<T> 返回類型。 它支持返回從 ActionResult 派生的類型或返回固定類型。ActionResult<T> 提供以下優勢:

     (1) 簡化ProducesResponseType  例如:[ProducesResponseType(200, Type = typeof(Product))] 可簡化為 [ProducesResponseType(200)]

     (2) 隱式強制轉換運算符,將 T 轉換為 ObjectResult,也就是將 return new ObjectResult(T); 簡化為 return T;

    下麵是同步示例,(1)簡化ProducesResponseType,(2)返回隱式轉換。

[HttpGet("{id}")]
[ProducesResponseType(StatusCodes.Status404NotFound)]
public ActionResult<Product> GetById(int id)
{
    if (!_repository.TryGetProduct(id, out var product))
    {
        return NotFound();
    }
   // return new ObjectResult(product);
    return product;
}

     下麵是非同步示例:

[HttpPost]
[ProducesResponseType(StatusCodes.Status201Created)]
[ProducesResponseType(StatusCodes.Status400BadRequest)]
public async Task<ActionResult<Product>> CreateAsync(Product product)
{
    if (product.Description.Contains("XYZ Widget"))
    {
        return BadRequest();
    }

    await _repository.AddProductAsync(product);

    return CreatedAtAction(nameof(GetById), new { id = product.Id }, product);
}

     適用場景:對比IActionResult類型適用場景,它提供了二種優勢。

    最後建議:不要用特定類型返回。 對於有返回類型的使用ActionResult<T>,相反對於沒有返回類型的使用IActionResult。 二者使用在“ asp.net core系列 36 WebAPI 搭建詳細示例”中有介紹。

 

二.響應數據的格式化

  響應數據是:response返回到客戶端的數據。在ASP.NET Core MVC 中,包含對固定格式(json,xml,string..)或根據客戶端規範(Accept)來設置響應數據格式的內置支持。預設返回json數據格式

 

  2.1 設置固定格式的action結果

    對於返回固定格式,例如返回JsonResult 和 ContentResult。這樣api向客戶端始終返回固定的格式,不考慮客戶端的Accept選項設置。JsonResult 始終返回josn數據格式, ContentResult始終返回純文本數據格式。如果不需要Action返回固定數據格式,可以返回IActionResult ,這樣可以有多種選擇的數據格式。預設是json數據格式。

    (1) 返回json格式的數據,使用fiddler請求url,返回客戶端json格式數據,如下圖:

        /// <summary>
        /// 返回固定的json格式字元串
        /// </summary>
        /// <returns></returns>
        [HttpGet("Get")]
        public JsonResult Get()
        {
            return Json(_context.TodoItems.ToList());
        }

    (2) 返回純文本格式數據,使用fiddler請求url,返回客戶端字元串,如下圖

        /// <summary>
        /// 返回固定的字元串格式
        /// </summary>
        /// <returns></returns>
        [HttpGet("Message")]
        public ContentResult Message()
        {
            return Content("hello");
        }

 

  2.2 返回格式協商

     在公開的api的場景,請求方(客戶端)在獲取數據時,他們可能要求返回自己想要的數據格式。這樣就不能使用固定的數據格式(一般也不推薦)。當客戶端指定 Accept 標頭時,就可以實現內容協商,對於內容協商返回數據格式由 ObjectResult 實現

     下麵的案例中,返回IActionResult,返回的數據格式,由ObjectResult 來確定,預設是json數據格式。

        [HttpGet("{id}", Name = "GetTodoItem")]
        public async Task<ActionResult<TodoItem>> GetTodoItem(long id)
        {
            var todoItem = await _context.TodoItems.FindAsync(id);
            if (todoItem == null)
            {
                //返回狀態碼404,打包到了ObjectResult中
                return NotFound();
            }

            //返回實體,打包到了ObjectResult中
            return todoItem;
        }

    客戶端通過指定Accept: application/xml,希望返回xml數據格式,但還是json數據格式(見下圖)。這是因為:

    (1) 預設情況下,當框架檢測到請求來自瀏覽器時,它將忽略 Accept 標頭轉而以應用程式的配置預設格式。

    (2) 如果請求指定 XML,但是未配置 XML 格式化程式,那麼將使用 JSON 格式化程式。

    如果應用程式要服從瀏覽器 accept 標頭,可以將此配置為 MVC 配置的一部分,方法是在 Startup.cs 中以 ConfigureServices 方法將 RespectBrowserAcceptHeader 設置為 true,並設置以客戶端格式優先。

            services.AddMvc(options =>
            {
                //優先客戶端指定數據格式
                options.RespectBrowserAcceptHeader = true;
                //添加xml數據格式的輸出
                options.OutputFormatters.Add(new XmlSerializerOutputFormatter());
            });

    如下圖所示,客戶端指定Accept: application/xml,服務端就返回了xml數據格式,同樣指定Accept: application/json ,服務端就返回json數據格式。

    

  2.3 強制執行固定格式

    如果需要限制固定Action的響應格式,那麼可以應用 [Produces] 篩選器。 [Produces] 篩選器指定特定action(或控制器)的響應格式。 如同大多篩選器,這可以在action層面、控制器層面或全局範圍內應用。 這樣格式協商就失敗,始終返回json數據格式

       //控制器層面強制使用json格式
      [Produces("application/json")]
      [ApiController]//添加特性,代表是一個Web API控制器
      public class TodoController : Controller

      //action層面強制返回json格式
       [Produces("application/json")]
       [HttpGet]
       public async Task<ActionResult<IEnumerable<TodoItem>>> GetTodoItems()
       {
            //using Microsoft.EntityFrameworkCore;
            return await _context.TodoItems.ToListAsync();
       }  

    

       2.4 特殊情況格式化程式

    如果要過濾客戶端Accept請求的某些類型,例如過濾text/plain。string 預設是text/plain類型,如果刪除TextOutputFormatter,則string返回類型 是406 Not Acceptable。

  //下麵對返回string字元串或返回http 204的進行過濾,代碼對應如下:
  services.AddMvc(options =>
  {
      options.OutputFormatters.RemoveType<TextOutputFormatter>();
      options.OutputFormatters.RemoveType<HttpNoContentOutputFormatter>();
  });

 

  2.5 響應格式URL映射

    當格式協商配置好了以後,客戶端可以請求特定格式作為URL的一部分。下麵是Url映射的配置示例。

  [FormatFilter]
  public class ProductsController
  {
      [Route("[controller]/[action]/{id}.{format?}")]
      public Product GetById(int id)

     當客戶端訪問該url,返回預設數據格式:

    /products/GetById/5

 

    當客戶端訪問該url,返回json數據格式:

    /products/GetById/5.json

 

    當客戶端訪問該url,返回xml數據格式:

    /products/GetById/5.xml

 

  參考文獻:    

    操作返回類型

    響應格式化

 


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

-Advertisement-
Play Games
更多相關文章
  • 本來計劃接著上篇 C# Winform模仿百度日曆,發現一時半會寫不完,只寫了一小半還不全,暫且擱置下。現在計划下班後每天至少寫一篇博客,未能完成的等周末(不加班都情況)補充完整。 本篇博客窗體換膚,不是本人原創是之前 程式員之窗的作品。我看界面挺好,當時特別特別菜(當然現在依然也很菜),對此佩服的... ...
  • 關於CAS是個什麼東西,就不多閑扯了,相信每個有過SSO經驗的都聽過CAS大名,百度百科地址: https://baike.baidu.com/item/CAS/1329561?fr=aladdin 項目所用https:/github.com/apereo/dotnet-cas-client 在.N ...
  • 前言 看了一百遍,不如動手寫一遍。 Socket這塊使用不是特別熟悉,之前實現是公司有對應源碼改改能用。 但是不理解實現的過程和步驟,然後最近有時間自己寫個demo實現看看,熟悉熟悉Socket。 網上也有好的文章,結合別人的理接和自己實踐總算寫完了。。。 參考: https://www.cnblo ...
  • 1.創建解決方案 2.選擇類型-Web API 3.設置項目生成XML路徑 同時修改HelpPageConfig,代碼調用XML文件位置 3.編寫WebApi介面代碼 4.啟動項目 查看介面 5.測試介面(添加WebApiTestClient組件) 安裝完成之後,修改Api.cshtml文件 這個時 ...
  • 二、傳統的委托 接下來講一講方法參數。下麵以“餐館服務員為客戶下單”[2]的事件作為描述。一般對事件的做法分3個部分: 1. 方法參數 EventArgs,一般用於傳送數據。在本例場景中 2 . 觸發事件的對象 // 下單的事件是Customer對象擁有的,∴寫在Customer類當中 3 . 執行 ...
  • 一、簡單的委托 1.1 委托的聲明: C#當中,委托(delegate)是一種方法封裝,也即委托對象可以作為一種傳遞方法的變數來使用。 委托也算是一種類,與類是平級的存在。在類中寫delegate對象當然是允許的,畢竟C#也允許類中類。但是一般不這樣做,委托的聲明最好與類聲明平級。 聲明: 方法執行 ...
  • 鏈接:https://pan.baidu.com/s/1pLzOlTv0nqSbhzujHZht1w 提取碼:1m9l AccessHelper: //Microsoft.ACE.OLEDB.12.0是連接access2007之後的資料庫使用的 //Microsoft.Jet.OLEDB.4.0是連 ...
  • 一、準備工作 1、使用虛擬機:VMware-workstation-full-15.0.2版本。 下載地址:http://download3.vmware.com/software/wkst/file/VMware-workstation-full-15.0.0-10134415.exe 2、秘鑰: ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...