ASP.NET Web API 2.0 统一响应格式
传统实现
在搭建 Web API 服务的时候,针对客户端请求,我们一般都会自定义响应的 JSON 格式,比如:
{
"Data" : {
"Id" : 100,
"Name" : "Robin"
},
"ErrorMessage" : "错误消息"
}
在基于 ASP.NET Web API 的应用程序,我们一般会创建一个相应结构的 C# 类,如下:
public class ApiResult
{
public string ErrorMessage { get; set; }
public object Data { get; set; }
}
这里约定, ErrorMessage 为空或null,即表示没有异常,这时 Data 就是需要的数据;反之如果 ErrorMessage 不为空或null, 则代表错误消息,这时 Data 为null。
接下来在 Action 中返回该类的一个实例, Web API 会在内部调用格式化器将对象序列化为 JSON 或 XML 等格式,如下:
public class UserController : ApiController
{
public IHttpActionResult GetUser()
{
return new ApiResult()
{
Data = new User{ Id = 100, Name = "Robin" },
ErrorMessage = string.Empty
};
}
}
public class User
{
public int Id {get; set;}
public string Name {get; set;}
}
好了,传统做法就是这样,也可以实现。但是再进一步考虑,如果有非常多的 Action 方法,每次都要写 reutrn new ApiResult(){......} 是不是特别繁琐呢?
问题
有没有办法在 Action 方法中只返回真正需要的数据,但是返回给客户端时又统一成约定的 JSON 结构呢?
解决方案
当然有办法,借助 Web API 提供的 ActionFilter 就可以实现。
首先我们新建一个 CustomActionFilter :
public class CustomActionFilter : ActionFilterAttribute
{
public override void OnActionExecuted(HttpActionExecutedContext context)
{
var content = context.Response?.Content as ObjectContent;
if (content != null)
{
content.Value = new ApiResult
{
Data = content.Value,
};
}
}
}
然后 Action 方法这样写:
public User GetAll()
{
return new new User{Id = 100, Name = "Robin"};
}
这样实现的另一个好处是,由于返回值是强类型,可以据此生成 API 文档,方法的可读性也更好。
异常处理
前面提到的需求实现了,然后再进一步考虑,如何处理异常情况?
如果由于代码 BUG 抛出未处理的异常,Web API仍然会调用 CustomActionFilter 中的代码,但是这时 Response = null,也就无法给 content.Value 重新赋值。
这时 Web API 会将框架约定的 JSON 消息返回给客户端,而不是我们业务上需要的,如下是 Web API 抛出的未处理异常消息:
{
"Message": "An error has occurred.",
"ExceptionMessage": "No MessageException parameter",
"ExceptionType": "Framework.Common.MessageException",
"StackTrace": " 在 Controllers.FooController.GetAll() 位置......
}
这时如果还希望异常消息遵循业务约定的 JSON 格式,该如何做呢?
这里要分几种情况:
Action 内的异常
可以直接在 CustomActionFilter 的 OnActionExecuted 方法中处理,改造后的代码如下:
public override void OnActionExecuted(HttpActionExecutedContext context)
{
var content = context.Response?.Content as ObjectContent;
if (content != null)
{
content.Value = new ApiResult { Data = content.Value };
}
// 设置发生异常时的消息
if (context.Exception != null)
{
context.Response = new HttpResponseMessage(HttpStatusCode.InternalServerError)
{
Content = new StringContent(JsonConvert.SerializeObject(
new ApiResult
{
ErrorMessage = context.Exception.Message
}), Encoding.UTF8, "application/json")
};
}
}
同样也可以用自定义 ExceptionFilter 来达到同样的目的,这里为了简化不再贴代码。
其他异常
ActionFilterAttribute 和 ExceptionFilterAttribute 都只能处理部分异常,比如 Action 内的异常,但是譬如 以下的几种未处理异常,过滤器就爱莫能助了:
- 来自 Controller 构造器的异常。
- 来自 Message Handlers 的异常。
- 匹配路由过程中的异常
- 在序列化响应内容期间产生的异常
为了处理全局范围内的未处理异常,Web API 提供了 ExceptionHandler 和 ExceptionLogger。
详情可以参考我翻译的文档:ASP.NET Web API 2 中的全局错误处理
其中只有 ExceptionHandler 可以在捕捉到未处理异常并处理后,对响应消息进行重新设置,而 ExceptionLogger 则不能。
代码如下:
public class CollectServiceExceptionHandler : ExceptionHandler
{
public override Task HandleAsync(ExceptionHandlerContext context, CancellationToken cancellationToken)
{
context.Result = new ApiResult { ErrorMessage = context.Exception.Message };
return base.HandleAsync(context, cancellationToken);
}
}
注意:这里 ExceptionHandlerContext 的 Result 属性的类型是 IHttpActionResult,所以**ApiResult ** 类要实现 IHttpActionResult 接口。
ExceptionHandler 的用途就是:接收全局范围内未处理的异常,然后返回一个自定义的错误消息。
总结
实现开篇的需求,有三种实现方式:
- 自定义 ActionFilterAttribute
- 自定义 ExceptionFilterAttribute
- 自定义 ExceptionHandler
补充:经 @ichengzi 指出,『web api 2.0 之前的版本不支持这种处理方法』。
ASP.NET Web API 2.0 统一响应格式的更多相关文章
- ASP.NET Web API 2.0新特性:Attribute Routing1
ASP.NET Web API 2.0新特性:Attribute Routing[上篇] 对于一个针对ASP.NET Web API的调用请求来说,请求的URL和对应的HTTP方法的组合最终决定了目标 ...
- 一个ASP.NET Web API 2.0应用
在一个空ASP.NET Web项目上创建一个ASP.NET Web API 2.0应用 由于ASP.NET Web API具有与ASP.NET MVC类似的编程方式,再加上目前市面上专门介绍ASP.N ...
- How ASP.NET Web API 2.0 Works?[持续更新中…]
一.概述 RESTful Web API [Web标准篇]RESTful Web API [设计篇] 在一个空ASP.NET Web项目上创建一个ASP.NET Web API 2.0应用 二.路由 ...
- 在一个空ASP.NET Web项目上创建一个ASP.NET Web API 2.0应用
由于ASP.NET Web API具有与ASP.NET MVC类似的编程方式,再加上目前市面上专门介绍ASP.NET Web API 的书籍少之又少(我们看到的相关内容往往是某本介绍ASP.NET M ...
- 让ASP.NET Web API支持POST纯文本格式(text/plain)的数据
今天在web api中遇到了这样一个问题,虽然api的参数类型是string,但只能接收post body中json格式的string,不能接收原始string. web api是这样定义的: pub ...
- ASP.NET WEB API 初探
本文初步介绍如何简单创建一个ASP.NET Web Api 程序. Web Api 顾名思义就是一个Api接口,客户端可调用此接口进行业务操作.此类应用与 ASP.NET web服务(即使用扩展名. ...
- 基于.Net Framework 4.0 Web API开发(3):ASP.NET Web APIs 异常的统一处理Attribute 和统一写Log 的Attribute的实现
概述: ASP.NET Web API 的好用使用过的都知道,没有复杂的配置文件,一个简单的ApiController加上需要的Action就能工作.但是项目,总有异常发生,本节就来谈谈API的异常 ...
- ASP.NET Web API 记录请求响应数据到日志的一个方法
原文:http://blog.bossma.cn/dotnet/asp-net-web-api-log-request-response/ ASP.NET Web API 记录请求响应数据到日志的一个 ...
- 在 ASP.NET Web API 中使用 Attribute 统一处理异常
并非所有的异常都需要 try-catch 进行重复的处理,这会导致大量的重复性代码,一旦后续系统出现异常处理机制的修改,随着代码量增多,修改也会变的更加困难. ASP.NET Web API 中特别增 ...
随机推荐
- java并发:Condition的应用
Condition类可以使线程等待,也可以唤醒线程.Condition类的await方法和Object类的wait方法等效Condition类的signal方法和Object类的notify方法等效C ...
- 【校招面试 之 C/C++】第6题 C++深拷贝与浅拷贝
1.两个的区别(1)在未定义显示拷贝构造函数的情况下,系统会调用默认的拷贝函数——即浅拷贝,它能够完成成员的一一复制.当数据成员中没有指针时,浅拷贝是可行的: 但当数据成员中有指针时,如果采用简单的浅 ...
- BZOJ 1345[BOI]序列问题 - 贪心 + 单调栈
题解 真的没有想到是单调栈啊. 回想起被单调栈支配的恐惧 最优情况一定是小的数去合并 尽量多的数,所以可以维护一个递减的单调栈. 如果加入的数比栈首小, 就直接推入栈. 如果加入的数大于等于栈首, 必 ...
- csv中文乱码
处理办法:https://jingyan.baidu.com/album/3c48dd3464b46ce10be3581f.html?picindex=2
- Jmeter将HTTP request报文体中的字符串转换为大写
<awd><client id='${__javaScript("${IndividualID}".toUpperCase())}'><member ...
- C语言 链表基本函数
#include <stdio.h> #include <malloc.h> typedef struct my_node mynode; struct my_node{ ...
- 内网IP和公网IP的区别
内网IP和公网IP的区别 什么是内网IP: 一些小型企业或者学校,通常都是申请一个固定的IP地址,然后通过IP共享(IP Sharing),使用整个公司或学校的机器都能够访问互联网.而 ...
- PTA第五次作业
#include<stdio.h> #include<math.h> int main () { int n,m,i,j,a; scanf("%d",&am ...
- js把一个数组插入到另一个数组的指定位置
var arr1 = ['a', 'b', 'c']; var arr2 = ['1', '2', '3']; // 把arr2 变成一个适合splice的数组(包含splice前2个参数的数组) a ...
- centos6.8下redis的安装和配置
centos6.8下redis的安装和配置 下载.安装 在redis官网可以获取到最新版本的redis 进入/usr/local/目录,执行如下命令 wget http://download.redi ...