.NET|.NET Core API之格式化输出对象OutputFormatter
相信大家在项目中都用过统一响应参数模板。
先声明一个响应模板类:
public class ResponseDto{public int code { get; set; }public string msg { get; set; }public object data { get; set; }}
再定义返回成功和失败的方法:
public IActionResult Success(object data){ return ......}public IActionResult Fail(string msg){ return ......}
在接口返回时统一调用:
[HttpGet]public IActionResult Get(){ var data = https://www.it610.com/article/new WeatherForecast() { Date = DateTime.Now }; return Success(data); }
文章图片
当然了,这篇文章所讲的
OutputFormatter
和上面的统一模板不冲突哈,存在共通之处,都是格式化响应参数嘛,拿来做个引子。OutputFormatter
OutputFormatter
是所有格式化输出的基类,有唯一的子类:TextOutputFormatter
,同时TextOutputFormatter
又有一大堆子类:- JsonOutputFormatter
- NewtonsoftJsonOutputFormatter
- StringOutputFormatter
- SystemTextJsonOutputFormatter
- XmlDataContractSerializerOutputFormatter
- XmlSerializerOutputFormatter
asp.net core api
响应参数默认的输出格式就是json
。猴:这个接口给我返回
xml
,我不要json
。我:你是不是脑子有毛病?好好的
json
不用用xml
。得,前端大佬得要求还是得满足不是,这时候有些同学是不是已经去百度:
.Net怎么将对象转换成xml?
No
No
No
,这时候就轮到OutputFormatter
的孙子 XmlDataContractSerializerOutputFormatter
出场了。只需要简单给接口配置一个属性就搞定啦。
[Produces("application/xml")][HttpGet]public WeatherForecast Get(){ return new WeatherForecast() { Date = DateTime.Now }; }
我们来运行看一看:
文章图片
wtf
,怎么会406
。406:表示客户端无法解析服务端返回的内容。说白了就是后台的返回结果前台无法解析就报406错误。
哦,原来是忘了在
Startup
中配置我们的孙子XmlDataContractSerializerOutputFormatter
。services.AddControllers((c) =>{ c.OutputFormatters.Add(new XmlDataContractSerializerOutputFormatter()); });
文章图片
注意:不只是没有在
Startup
中会出现406
哦,以下情况也会出现:- contentType不存在
- contentType与响应参数不匹配
上面介绍了内置
OutputFormatter
的使用,那如果我们想自定义呢?当然也是可以的。下面我们就用自定义的
OutputFormatter
实现顶部响应模板的效果:public class ObjectOutputFormatter : TextOutputFormatter{ public ObjectOutputFormatter() {SupportedEncodings.Add(Encoding.UTF8); SupportedEncodings.Add(Encoding.Unicode); // 这就是我们自定义contentType的名称SupportedMediaTypes.Add("text/object"); } public override async Task WriteResponseBodyAsync(OutputFormatterWriteContext context, Encoding selectedEncoding) {if (context == null){throw new ArgumentNullException(nameof(context)); }if (selectedEncoding == null){throw new ArgumentNullException(nameof(selectedEncoding)); }string text = JsonConvert.SerializeObject(new ResponseDto(){msg = "成功,自定义的哦",code = 200,data = https://www.it610.com/article/context.Object}); var response = context.HttpContext.Response; await response.WriteAsync(text, selectedEncoding); }}[Produces("text/object")][HttpGet]public WeatherForecast Get(){ return new WeatherForecast() { Date = DateTime.Now }; }public void ConfigureServices(IServiceCollection services){ services.AddControllers((c) => {c.OutputFormatters.Add(new XmlDataContractSerializerOutputFormatter()); // 我们自定义的输出格式c.OutputFormatters.Add(new ObjectOutputFormatter()); }); }
搞定,我们来看看效果:
文章图片
ActionFilterAttribute
有些同学可能会想到过滤器,是的,上面的效果过滤器也能实现:
public class ResultFilter : ActionFilterAttribute{public override void OnResultExecuting(ResultExecutingContext context){ResponseDto result = new ResponseDto(); result.code = 200; result.msg = "成功,ResultFilter"; var properties = context.Result.GetType().GetProperties(BindingFlags.Instance | BindingFlags.Public); result.data = https://www.it610.com/article/properties.FirstOrDefault(c => c.Name == "Value").GetValue(context.Result); context.Result = new JsonResult(result); base.OnResultExecuting(context); }}[TypeFilter(typeof(ResultFilter))][HttpGet]public WeatherForecast Get(){ return new WeatherForecast() { Date = DateTime.Now }; }
文章图片
猴:有了过滤器为什么还搞个
OutputFormatter
呢?我:不能因为过滤器可以实现同样的功能就认为
OutputFormatter
多余了,很显然过滤器的操作对象是请求/响应上下文,而OutputFormatter
的操作对象则是响应参数。再说了,ActionFilterAttribute
过滤器只是众多过滤器的一种。猴:那过滤器和自定义
OutputFormatter
一起用会是什么效果呢?是不是像下面这样?文章图片
我:不是,过滤器和自定义
OutputFormatter
同时使用,生效的只有过滤器,不信可以打断点试一下哦。[Produces("text/object")][TypeFilter(typeof(ResultFilter))][HttpGet]public WeatherForecast Get(){ return new WeatherForecast() { Date = DateTime.Now }; }
文章图片
【.NET|.NET Core API之格式化输出对象OutputFormatter】到此这篇关于.NET Core API之格式化输出对象OutputFormatter的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持脚本之家。
推荐阅读
- Android 音视频开发(使用 Camera API 采集视频数据)
- JSTL Core c:set标记
- JSTL Core c:remove标记
- JSTL Core c:out标签
- 什么是gcapi.dll、用途以及如何删除它()
- React-Native 问题随记2( com.android.builder.testing.api.DeviceException)
- JSP API介绍
- Android API的一点了解
- 爆赞 Apipost V6.0.4版本 支持离线使用
- Win8下如何让iis支持asp、asp.net?