網站首頁 編程語言 正文
相信大家在項目中都用過統一響應參數模板。
先聲明一個響應模板類:
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 = 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
的使用,那如果我們想自定義呢?當然也是可以的。
下面我們就用自定義的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 = 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 = 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 }; }
原文鏈接:https://www.cnblogs.com/cool-net/p/16176643.html
相關推薦
- 2023-10-15 AddressSanitizer 查找內存問題
- 2022-09-10 Python遞歸實現猴子吃桃問題及解析_python
- 2022-01-08 解決npm install報錯問題--npm install xxx npm ERR! code E
- 2023-10-17 關于for循環遍歷不同表單校驗的bug(需要多次校驗)
- 2022-04-14 如何解決:git push error: failed to push some refs to
- 2022-03-23 C++實現水仙花數判斷實例_C 語言
- 2022-04-02 在python3中使用Supervisor的詳細教程_python
- 2022-06-08 兩步配置解決 IDEA新項目maven依賴問題
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支