热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

asp.netcore系列38WebAPI返回类型与响应格式--必备

原文:asp.netcore系列38WebAPI返回类型与响应格式--必备一.返回类型ASP.NETCore提供以下WebAPIAction方法返回类型选项,以及说明每
原文: asp.net core系列 38 WebAPI 返回类型与响应格式--必备

一.返回类型

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

  (1) 固定类型

  (2) IActionResult

  (3) ActionResult

  

  1.1 固定类型

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

[HttpGet]
public IEnumerable 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 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

     ASP.NET Core 2.1 引入了 ActionResult 返回类型。 它支持返回从 ActionResult 派生的类型或返回固定类型。ActionResult 提供以下优势:

     (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 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> 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,相反对于没有返回类型的使用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格式数据,如下图:

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

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

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

 

  2.2 返回格式协商

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

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

        [HttpGet("{id}", Name = "GetTodoItem")]
        public async Task> 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>> 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();
      options.OutputFormatters.RemoveType();
  });

 

  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

 

  参考文献:    

    操作返回类型

    响应格式化

 


推荐阅读
  • 本文详细介绍了Java中org.neo4j.helpers.collection.Iterators.single()方法的功能、使用场景及代码示例,帮助开发者更好地理解和应用该方法。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 本文详细介绍了Java中org.eclipse.ui.forms.widgets.ExpandableComposite类的addExpansionListener()方法,并提供了多个实际代码示例,帮助开发者更好地理解和使用该方法。这些示例来源于多个知名开源项目,具有很高的参考价值。 ... [详细]
  • 深入解析Spring Cloud Ribbon负载均衡机制
    本文详细介绍了Spring Cloud中的Ribbon组件如何实现服务调用的负载均衡。通过分析其工作原理、源码结构及配置方式,帮助读者理解Ribbon在分布式系统中的重要作用。 ... [详细]
  • 在前两篇文章中,我们探讨了 ControllerDescriptor 和 ActionDescriptor 这两个描述对象,分别对应控制器和操作方法。本文将基于 MVC3 源码进一步分析 ParameterDescriptor,即用于描述 Action 方法参数的对象,并详细介绍其工作原理。 ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • 本文详细介绍了如何构建一个高效的UI管理系统,集中处理UI页面的打开、关闭、层级管理和页面跳转等问题。通过UIManager统一管理外部切换逻辑,实现功能逻辑分散化和代码复用,支持多人协作开发。 ... [详细]
  • 本文介绍如何在 Android 中通过代码模拟用户的点击和滑动操作,包括参数说明、事件生成及处理逻辑。详细解析了视图(View)对象、坐标偏移量以及不同类型的滑动方式。 ... [详细]
  • 深入解析Android自定义View面试题
    本文探讨了Android Launcher开发中自定义View的重要性,并通过一道经典的面试题,帮助开发者更好地理解自定义View的实现细节。文章不仅涵盖了基础知识,还提供了实际操作建议。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 本文详细介绍如何使用Python进行配置文件的读写操作,涵盖常见的配置文件格式(如INI、JSON、TOML和YAML),并提供具体的代码示例。 ... [详细]
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • 本文基于刘洪波老师的《英文词根词缀精讲》,深入探讨了多个重要词根词缀的起源及其相关词汇,帮助读者更好地理解和记忆英语单词。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
author-avatar
呼呼小琰琰
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有