一.返回类型

  ASP.NET Core 提供以下 Web API Action方法返回类型选项,以及说明每种返回类型的最佳适用情况:

  (1) 固定类型

  (2) IActionResult

  (3) ActionResult<T>

  

  1.1 固定类型

    最简单的操作是返回基元或复杂数据类型(如 string 或自定义对象类型)。 请参考以下Action,该Action返回自定义 Product 对象的集合:

  1. [HttpGet]
  2. public IEnumerable<Product> Get()
  3. {
  4. return _repository.GetProducts();
  5. }

    适用场景:在执行Action期间,无需要根据条件判断返回不同类型,只返回固定类型即可满足要求。 上述操作不接受任何参数,因此不需要参数约束验证。

  1.2  IActionResult类型

    当Action方法中可能有多个 ActionResult 返回类型时,适合使用 IActionResult 返回类型。ActionResult 类型可以表示多种 HTTP 状态代码。 属于此类别的一些常见返回类型包括:BadRequestResult (400)、NotFoundResult (404) 和 OkObjectResult (200)。

    由于Action方法中有多个返回类型和路径,因此必须使用 [ProducesResponseType] 特性。 此特性可针对 Swagger 等工具生成的 API 帮助页生成更多描述性响应详细信息(上篇有介绍)。 [ProducesResponseType] 指示Action将返回的已知类型和 HTTP 状态代码。

    下面是一个同步Action,该Action方法中可能有两种返回类型:

  1. [HttpGet("{id}")]
  2. [ProducesResponseType(typeof(Product), StatusCodes.Status200OK)]
  3. [ProducesResponseType(StatusCodes.Status404NotFound)]
  4. public IActionResult GetById(int id)
  5. {
  6. if (!_repository.TryGetProduct(id, out var product))
  7. {
  8. return NotFound();
  9. }
  10.  
  11. return Ok(product);
  12. }

    下面是一个异步Action,该Action方法中可能有两种返回类型:

  1. [HttpPost]
  2. [ProducesResponseType(typeof(Product), StatusCodes.Status201Created)]
  3. [ProducesResponseType(StatusCodes.Status400BadRequest)]
  4. public async Task<IActionResult> CreateAsync([FromBody] Product product)
  5. {
  6. if (product.Description.Contains("XYZ Widget"))
  7. {
  8. return BadRequest();
  9. }
  10.  
  11. await _repository.AddProductAsync(product);
  12.  
  13. return CreatedAtAction(nameof(GetById), new { id = product.Id }, product);
  14. }

    适用场景:当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)返回隐式转换。

  1. [HttpGet("{id}")]
  2. [ProducesResponseType(StatusCodes.Status404NotFound)]
  3. public ActionResult<Product> GetById(int id)
  4. {
  5. if (!_repository.TryGetProduct(id, out var product))
  6. {
  7. return NotFound();
  8. }
  9. // return new ObjectResult(product);
  10. return product;
  11. }

    下面是异步示例:

  1. [HttpPost]
  2. [ProducesResponseType(StatusCodes.Status201Created)]
  3. [ProducesResponseType(StatusCodes.Status400BadRequest)]
  4. public async Task<ActionResult<Product>> CreateAsync(Product product)
  5. {
  6. if (product.Description.Contains("XYZ Widget"))
  7. {
  8. return BadRequest();
  9. }
  10.  
  11. await _repository.AddProductAsync(product);
  12.  
  13. return CreatedAtAction(nameof(GetById), new { id = product.Id }, product);
  14. }

    适用场景:对比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格式数据,如下图:

  1. /// <summary>
  2. /// 返回固定的json格式字符串
  3. /// </summary>
  4. /// <returns></returns>
  5. [HttpGet("Get")]
  6. public JsonResult Get()
  7. {
  8. return Json(_context.TodoItems.ToList());
  9. }

    (2) 返回纯文本格式数据,使用fiddler请求url,返回客户端字符串,如下图

  1. /// <summary>
  2. /// 返回固定的字符串格式
  3. /// </summary>
  4. /// <returns></returns>
  5. [HttpGet("Message")]
  6. public ContentResult Message()
  7. {
  8. return Content("hello");
  9. }

  2.2 返回格式协商

    在公开的api的场景,请求方(客户端)在获取数据时,他们可能要求返回自己想要的数据格式。这样就不能使用固定的数据格式(一般也不推荐)。当客户端指定 Accept 标头时,就可以实现内容协商,对于内容协商返回数据格式由 ObjectResult 实现 。

    下面的案例中,返回IActionResult,返回的数据格式,由ObjectResult 来确定,默认是json数据格式。

  1. [HttpGet("{id}", Name = "GetTodoItem")]
  2. public async Task<ActionResult<TodoItem>> GetTodoItem(long id)
  3. {
  4. var todoItem = await _context.TodoItems.FindAsync(id);
  5. if (todoItem == null)
  6. {
  7. //返回状态码404,打包到了ObjectResult中
  8. return NotFound();
  9. }
  10.  
  11. //返回实体,打包到了ObjectResult中
  12. return todoItem;
  13. }

    客户端通过指定Accept: application/xml,希望返回xml数据格式,但还是json数据格式(见下图)。这是因为:

    (1) 默认情况下,当框架检测到请求来自浏览器时,它将忽略 Accept 标头转而以应用程序的配置默认格式。

    (2) 如果请求指定 XML,但是未配置 XML 格式化程序,那么将使用 JSON 格式化程序。

    如果应用程序要服从浏览器 accept 标头,可以将此配置为 MVC 配置的一部分,方法是在 Startup.cs 中以 ConfigureServices 方法将 RespectBrowserAcceptHeader 设置为 true,并设置以客户端格式优先。

  1. services.AddMvc(options =>
  2. {
  3. //优先客户端指定数据格式
  4. options.RespectBrowserAcceptHeader = true;
  5. //添加xml数据格式的输出
  6. options.OutputFormatters.Add(new XmlSerializerOutputFormatter());
  7. });

    如下图所示,客户端指定Accept: application/xml,服务端就返回了xml数据格式,同样指定Accept: application/json ,服务端就返回json数据格式。

    

  2.3 强制执行固定格式

    如果需要限制固定Action的响应格式,那么可以应用 [Produces] 筛选器。 [Produces] 筛选器指定特定action(或控制器)的响应格式。 如同大多筛选器,这可以在action层面、控制器层面或全局范围内应用。 这样格式协商就失败,始终返回json数据格式。

  1. //控制器层面强制使用json格式
  2. [Produces("application/json")]
  3. [ApiController]//添加特性,代表是一个Web API控制器
  4. public class TodoController : Controller
  5.  
  6. //action层面强制返回json格式
  7. [Produces("application/json")]
  8. [HttpGet]
  9. public async Task<ActionResult<IEnumerable<TodoItem>>> GetTodoItems()
  10. {
  11. //using Microsoft.EntityFrameworkCore;
  12. return await _context.TodoItems.ToListAsync();
  13. }

    

 2.4 特殊情况格式化程序

    如果要过滤客户端Accept请求的某些类型,例如过滤text/plain。string 默认是text/plain类型,如果删除TextOutputFormatter,则string返回类型 是406 Not Acceptable。

  1.   //下面对返回string字符串或返回http 204的进行过滤,代码对应如下:
  2.   services.AddMvc(options =>
  3.   {
  4.   options.OutputFormatters.RemoveType<TextOutputFormatter>();
  5.   options.OutputFormatters.RemoveType<HttpNoContentOutputFormatter>();
  6.   });

  2.5 响应格式URL映射

    当格式协商配置好了以后,客户端可以请求特定格式作为URL的一部分。下面是Url映射的配置示例。

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

    当客户端访问该url,返回默认数据格式:

    /products/GetById/5

    当客户端访问该url,返回json数据格式:

    /products/GetById/5.json

    当客户端访问该url,返回xml数据格式:

    /products/GetById/5.xml

  参考文献:    

    操作返回类型

    响应格式化

asp.net core系列 38 WebAPI 返回类型与响应格式--必备的更多相关文章

  1. asp.net core系列 36 WebAPI 搭建详细示例

    一.概述 HTTP不仅仅用于提供网页.HTTP也是构建公开服务和数据的API强大平台.HTTP简单灵活且无处不在.几乎任何你能想到的平台都有一个HTTP库,因此HTTP服务可以覆盖广泛的客户端,包括浏 ...

  2. asp.net core系列 37 WebAPI 使用OpenAPI (swagger)中间件

    一.概述 在使用Web API时,对于开发人员来说,了解其各种方法可能是一项挑战.在ASP.NET Core上,Web api 辅助工具介绍二个中间件,包括:Swashbuckle和NSwag .NE ...

  3. asp.net core 系列之webapi集成Dapper的简单操作教程

    Dapper也是是一种ORM框架 这里记录下,使用ASP.NET 集成 Dapper 的过程,方便自己查看 至于Dapper的特性以及操作可以参考Dapper官方文档 1.创建数据库相关 在Sql S ...

  4. asp.net core 系列之webapi集成EFCore的简单操作教程

    因为官网asp.net core webapi教程部分,给出的是使用内存中的数据即 UseInMemoryDatabase 的方式, 这里记录一下,使用SQL Server数据库的方式即 UseSql ...

  5. asp.net core系列 77 webapi响应压缩

    一.介绍 背景:目前在开发一个爬虫框架,使用了.net core webapi接口作为爬虫调用入口,在调用 webapi时发现爬虫耗时很短(1秒左右),但客户端获取响应时间却在3~4秒.对于这个问题考 ...

  6. asp.net core系列 WebAPI 作者:懒懒的程序员一枚

    asp.net core系列 36 WebAPI 搭建详细示例一.概述1.1 创建web项目1.2 添加模型类1.3 添加数据库上下文1.4 注册上下文1.5 添加控制器1.6 添加Get方法1.7 ...

  7. 【目录】asp.net core系列篇

    随笔分类 - asp.net core系列篇 asp.net core系列 68 Filter管道过滤器 摘要: 一.概述 本篇详细了解一下asp.net core filters,filter叫&q ...

  8. WPF中的常用布局 栈的实现 一个关于素数的神奇性质 C# defualt关键字默认值用法 接口通俗理解 C# Json序列化和反序列化 ASP.NET CORE系列【五】webapi整理以及RESTful风格化

    WPF中的常用布局   一 写在开头1.1 写在开头微软是一家伟大的公司.评价一门技术的好坏得看具体的需求,没有哪门技术是面面俱到地好,应该抛弃对微软和微软的技术的偏见. 1.2 本文内容本文主要内容 ...

  9. ASP.NET Core 2.2 WebApi 系列【九】使用SignalR (作者:tenghao510 ) 学习及内容补充

    原文地址:  ASP.NET Core 2.2 WebApi 系列[九]使用SignalR 今天,看到了大牛的这篇博文,  发了一下评论, 我很惊喜, 没想到他很快就回复了我,  而且通过QQ帮助了S ...

随机推荐

  1. Android Studio升级到3.4遇到的问题总结

    1.gradle需要升级. 1).project的build.gradle文件写下如下代码: buildscript { repositories { google() jcenter() } dep ...

  2. selenium操作浏览器的前进和后退

    前进关键字:driver.forward() 后退关键字:driver.back() 测试对象:1.https://www.baidu.com/ 2.https://www.sogou.com/ 实例 ...

  3. 《Java项目中classpath路径详解》

    项目里用到了classpath路径来引用文件,那么classpath指的是哪里呢 我首先把上面的applicationContext.xml文件放在了src目录下发现可以. 那么classpath到底 ...

  4. python可视化库 Matplotlib 00 画制简单图像

    1.下载方式:直接下载Andaconda,简单快捷,减少准备环境的时间 2.图像 3.代码:可直接运行(有详细注释) # -*- encoding:utf-8 -*- # Copyright (c) ...

  5. Javascript 标识符及同名标识符的优先级

    一.定义 标识符(Identifier)就是一个名字,用来对变量.函数.属性.参数进行命名,或者用做某些循环语句中的跳转位置的标记. //变量 var Identifier = 123; //属性 ( ...

  6. 使用Ant Build时提示错误: 编码GBK的不可映射字符

    这个build.xml是由eclipse neon 2016.6生成的 我的情况是,所有文件都使用了UTF-8编码,build.xml第一行也好好写着UTF-8,但build时仍然有乱码,并且提示失败 ...

  7. gridlayout代码注释

    <div class="wrapper"> //定义一节或者一部分区域,它的css样式对应的css中class选择器的wrapper <div class=&qu ...

  8. Linux指令 压缩与解压

    打包: 格式:tar -cvf  压缩后的名称.tar   压缩的文件1 压缩的文件2 ```压缩的文件n(压缩多个文件为一份时各个文件以空格隔开) 例子:tar -cvf  tomcats.tar ...

  9. Edge-assisted Traffic Engineering and applications in the IoT

    物联网中边缘辅助的流量工程和应用 本文为SIGCOMM 2018 Workshop (Mobile Edge Communications, MECOMM)论文. 笔者翻译了该论文.由于时间仓促,且笔 ...

  10. FFmpeg 结构体学习(六): AVCodecContext 分析

    在上文FFmpeg 结构体学习(五): AVCodec 分析我们学习了AVCodec结构体的相关内容.本文,我们将讲述一下AVCodecContext. AVCodecContext是包含变量较多的结 ...