日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

如何在?ASP.NET?Core?Web?API?中處理?Patch?請求

瀏覽:68日期:2022-06-14 15:06:01
目錄一、概述二、將 Keys 保存在 Input Model 中三、定義 ModelBinderFactory 和 ModelBinder四、在 ASP.NET Core 項目中替換 ModelBinderFactory五、定義 AutoMapper 的 TypeConverter六、模型映射七、測試源碼參考資料一、概述

PUT 和 PATCH 方法用于更新現有資源。 它們之間的區別是,PUT 會替換整個資源,而 PATCH 僅指定更改。

在 ASP.NET Core Web API 中,由于 C# 是一種靜態語言(dynamic 在此不表),當我們定義了一個類型用于接收 HTTP Patch 請求參數的時候,在 Action 中無法直接從實例中得知客戶端提供了哪些參數。

比如定義一個輸入模型和數據庫實體:

public class PersonInput{ public string? Name { get; set; } public int? Age { get; set; } public string? Gender { get; set; }}public class PersonEntity{ public string Name { get; set; } public int Age { get; set; } public string Gender { get; set; }}

再定義一個以 FromForm 形式接收參數的 Action:

[HttpPatch][Route('patch')]public ActionResult Patch([FromForm] PersonInput input){ // 測試代碼暫時將 AutoMapper 配置放在方法內。 var config = new MapperConfiguration(cfg => {cfg.CreateMap<PersonInput, PersonEntity>()); }); var mapper = config.CreateMapper(); // entity 從數據庫讀取,這里僅演示。 var entity = new PersonEntity {Name = '姓名', // 可能會被改變Age = 18, // 可能會被改變Gender = '我可能會被改變', }; // 如果客戶端只輸入 Name 字段,entity 的 Age 和 Gender 將不能被正確映射或被置為 null。 mapper.Map(input, entity); return Ok();}curl --location --request PATCH 'http://localhost:5094/test/patch' \--form 'Name='foo''

如果客戶端只提供了 Name 而沒有其他參數,從 HttpContext.Request.Form.Keys 可以得知這一點。如果不使用 AutoMapper,那么接下來是丑陋的判斷:

var keys = _httpContextAccessor.HttpContext.Request.Form.Keys;if(keys.Contains('Name')){ // 更新 Name(這里忽略合法性判斷) entity.Name = input.Name!;}if (keys.Contains('Age')){ // 更新 Age(這里忽略合法性判斷) entity.Age = input.Age!;}// ...

本文提供一種方式來簡化這個步驟。

二、將 Keys 保存在 Input Model 中

定義一個名為 PatchInput 的類:

public abstract class PatchInput{ [BindNever] public ICollection<string>? PatchKeys { get; set; }}

PatchKeys 屬性不由客戶端提供,不參與默認綁定。

PersonInput 繼承自 PatchInput:

public class PersonInput : PatchInput{ public string? Name { get; set; } public int? Age { get; set; } public string? Gender { get; set; }}三、定義 ModelBinderFactory 和 ModelBinderpublic class PatchModelBinder : IModelBinder{ private readonly IModelBinder _internalModelBinder; public PatchModelBinder(IModelBinder internalModelBinder) {_internalModelBinder = internalModelBinder; } public async Task BindModelAsync(ModelBindingContext bindingContext) {await _internalModelBinder.BindModelAsync(bindingContext);if (bindingContext.Model is PatchInput model){ // 將 Form 中的 Keys 保存在 PatchKeys 中 model.PatchKeys = bindingContext.HttpContext.Request.Form.Keys;} }}public class PatchModelBinderFactory : IModelBinderFactory{ private ModelBinderFactory _modelBinderFactory; public PatchModelBinderFactory(IModelMetadataProvider metadataProvider,IOptions<MvcOptions> options,IServiceProvider serviceProvider) {_modelBinderFactory = new ModelBinderFactory(metadataProvider, options, serviceProvider); } public IModelBinder CreateBinder(ModelBinderFactoryContext context) {var modelBinder = _modelBinderFactory.CreateBinder(context);// ComplexObjectModelBinder 是 internal 類if (typeof(PatchInput).IsAssignableFrom(context.Metadata.ModelType) && modelBinder.GetType().ToString().EndsWith('ComplexObjectModelBinder')){ modelBinder = new PatchModelBinder(modelBinder);}return modelBinder; }}四、在 ASP.NET Core 項目中替換 ModelBinderFactoryvar builder = WebApplication.CreateBuilder(args);// Add services to the container.builder.Services.AddPatchMapper();

AddPatchMapper 是一個簡單的擴展方法:

public static class PatchMapperExtensions{ public static IServiceCollection AddPatchMapper(this IServiceCollection services) {services.Replace(ServiceDescriptor.Singleton<IModelBinderFactory, PatchModelBinderFactory>());return services; }}

到目前為止,在 Action 中已經能獲取到請求的 Key 了。

[HttpPatch][Route('patch')]public ActionResult Patch([FromForm] PersonInput input){ // 不需要手工給 input.PatchKeys 賦值。 return Ok();}

PatchKeys 的作用是利用 AutoMapper。

五、定義 AutoMapper 的 TypeConverterpublic class PatchConverter<T> : ITypeConverter<PatchInput, T> where T : new(){ /// <inheritdoc /> public T Convert(PatchInput source, T destination, ResolutionContext context) {destination ??= new T();var sourceType = source.GetType();var destinationType = typeof(T);foreach (var key in source.PatchKeys ?? Enumerable.Empty<string>()){ var sourcePropertyInfo = sourceType.GetProperty(key, BindingFlags.IgnoreCase | BindingFlags.Public | BindingFlags.Instance); if (sourcePropertyInfo != null) {var destinationPropertyInfo = destinationType.GetProperty(key, BindingFlags.IgnoreCase | BindingFlags.Public | BindingFlags.Instance);if (destinationPropertyInfo != null){ var sourceValue = sourcePropertyInfo.GetValue(source); destinationPropertyInfo.SetValue(destination, sourceValue);} }}return destination; }}

上述代碼可用其他手段來代替反射。

六、模型映射[HttpPatch][Route('patch')]public ActionResult Patch([FromForm] PersonInput input){ // 1. 目前僅支持 `FromForm`,即 `x-www-form_urlencoded` 和 `form-data`;暫不支持 `FromBody` 如 `raw` 等。 // 2. 使用 ModelBinderFractory 創建 ModelBinder 而不是 ModelBinderProvider 以便于未來支持更多的輸入格式。 // 3. 目前還沒有支持多級結構。 // 4. 測試代碼暫時將 AutoMapper 配置放在方法內。 var config = new MapperConfiguration(cfg => {cfg.CreateMap<PersonInput, PersonEntity>().ConvertUsing(new PatchConverter<PersonEntity>()); }); var mapper = config.CreateMapper(); // PersonEntity 有 3 個屬性,客戶端如果提供的參數參數不足 3 個,在 Map 時未提供參數的屬性值不會被改變。 var entity = new PersonEntity {Name = '姓名',Age = 18,Gender = '如果客戶端沒有提供本參數,那我的值不會被改變' }; mapper.Map(input, entity); return Ok();}七、測試curl --location --request PATCH 'http://localhost:5094/test/patch' \--form 'Name='foo''

curl --location --request PATCH 'http://localhost:5094/test/patch' \--header 'Content-Type: application/x-www-form-urlencoded' \--data-urlencode 'Name=foo'源碼

Tubumu.PatchMapper

支持 FromForm,即 x-www-form_urlencoded 和 form-data。支持 FromBody 如 raw 等。支持多級結構。參考資料

GraphQL.NET

如何在 ASP.NET Core Web API 中處理 JSON Patch 請求

到此這篇關于在 ASP.NET Core Web API 中處理 Patch 請求的文章就介紹到這了,更多相關ASP.NET Core Web API 處理 Patch 請求內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: ASP.NET
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
九色精品91| 一区免费在线| 亚洲欧美日韩一区在线观看| 激情综合婷婷| 国产欧美在线观看免费| 亚久久调教视频| 亚洲日本免费电影| 丝袜亚洲精品中文字幕一区| 久久久9色精品国产一区二区三区| 日韩和欧美一区二区三区| 日韩电影免费网站| 久久伦理在线| 性欧美长视频| 日韩精品亚洲专区在线观看| 久久国产小视频| 久久久久美女| 国产精品普通话对白| 视频一区中文字幕| 18国产精品| 精品视频在线观看网站| 国产精品不卡| 久久精品国产68国产精品亚洲| 精品在线播放| 中文字幕日韩亚洲| 石原莉奈在线亚洲三区| 日韩在线一二三区| 91九色综合| 日韩一区二区在线免费| 好看的亚洲午夜视频在线| 日本中文字幕不卡| 久久精品一区二区三区中文字幕| 久久uomeier| 亚洲欧美日韩专区| 亚洲啊v在线免费视频| 电影91久久久| 国产亚洲高清视频| 久久精品资源| 蜜臀久久99精品久久久久宅男| 国产日韩1区| re久久精品视频| 亚洲一区有码| a天堂资源在线| 亚洲欧美网站在线观看| 国产精品久久久久蜜臀| 亚洲欧美视频| 国际精品欧美精品| 在线视频亚洲欧美中文| 成人午夜网址| 亚洲毛片一区| 久久美女精品| 日韩av在线播放网址| 蜜臀a∨国产成人精品| 国产精品白丝久久av网站| 久久亚洲电影| 午夜欧美在线| 国产麻豆一区二区三区| 免费欧美在线视频| 欧美亚洲国产激情| 成人日韩在线| 日韩1区在线| 麻豆国产精品一区二区三区| 国产一区调教| 日韩国产在线观看| 亚洲在线网站| 狠狠干成人综合网| 成人羞羞视频在线看网址| 欧美一区自拍| 日韩欧美三区| 四虎精品一区二区免费| 国产毛片一区| 99精品电影| 久久免费国产| 在线成人直播| 亚洲一区中文| 免费在线成人| 99亚洲精品| 亚洲神马久久| 亚洲深爱激情| 久久精品亚洲| 精品黄色一级片| 久久久久久久欧美精品| 99香蕉国产精品偷在线观看| 午夜欧美精品| 亚洲精品在线二区| 国产欧美日韩一级| 免费一区二区三区在线视频| 久久av在线| 久久国产视频网| 久久精品理论片| 日韩精品一卡| 日本亚洲不卡| 精品亚洲自拍| 欧美在线影院| 欧美在线黄色| 日韩免费在线| 综合亚洲色图| 国产精品密蕾丝视频下载| 国产精选在线| 在线视频亚洲欧美中文| 久久爱www.| 亚洲手机视频| 99香蕉国产精品偷在线观看| 日本中文字幕一区二区| 日韩免费视频| 91伊人久久| 亚洲国内欧美| 欧美日韩一区二区国产| 欧洲精品一区二区三区| 亚洲综合中文| 成人三级高清视频在线看| 亚洲欧美久久精品| 欧美日韩精品一区二区视频| 亚洲区第一页| 成人精品亚洲| 精品视频一区二区三区在线观看 | 久久福利影视| 亚洲不卡av不卡一区二区| 日本视频一区二区| 国产毛片久久| 中文字幕系列一区| 国内不卡的一区二区三区中文字幕| 亚洲精品小说| 婷婷综合六月| 韩日一区二区| 欧美精品中文| 婷婷综合福利| 婷婷亚洲综合| 久久影院一区| 狠狠躁少妇一区二区三区| 美女久久久久久| 国产精品一区二区精品视频观看| 日韩午夜av在线| 一区免费视频| 在线视频亚洲| 欧美一区=区| 99国产精品99久久久久久粉嫩| 久久精品国产亚洲夜色av网站| 日韩一区二区三区免费| 久久精品影视| 在线 亚洲欧美在线综合一区| 99精品小视频| 91亚洲一区| 精品福利久久久| 久久精品免费看| 91中文字幕精品永久在线| 欧美三级精品| 久久福利影视| 国产日产一区| yellow在线观看网址| 日韩在线视频精品| 五月天综合网站| 三级久久三级久久久| 国产日本精品| 免费在线小视频| 亚洲综合二区| 国产亚洲观看| 欧美韩日一区| 久久一区二区三区电影| 天使萌一区二区三区免费观看| 日韩毛片网站| 青青久久av| 日韩高清中文字幕一区| 精品视频一区二区三区在线观看| 国产aⅴ精品一区二区四区| 欧美日韩水蜜桃| 日韩国产在线观看一区| 日韩专区精品| 美美哒免费高清在线观看视频一区二区| 日本视频一区二区| 99久久视频| 国产精品久久久久久久久久久久久久久 | 免费看精品久久片| 国产精品久久久免费| 国产日韩专区| 国产成人精品三级高清久久91| 免费日韩一区二区| 日韩久久一区二区三区| 91亚洲精品在看在线观看高清| 日韩精品第一区| 亚洲欧美日本国产| 在线天堂资源www在线污| 日本不卡在线视频| 一区在线观看| 99精品视频在线| 久久久久久久久久久妇女| 蜜臀久久久99精品久久久久久| 日本在线高清| 精品视频黄色| 国产精品一页| 欧美久久精品| 国产日韩欧美一区二区三区| 九九在线精品| 天堂中文av在线资源库| 国产精品宾馆| 国产精品一区二区三区美女 | 视频一区在线播放| 蜜臀91精品一区二区三区| 亚洲有吗中文字幕| 中文字幕成人| 亚洲欧洲另类|