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

ASP.NETCore的ModelBinding来源与校验的介绍

ASP.NETCoreMVC的ModelBinding会将HTTPRequest数据,以映射的方式对应到参数中。基本上跟ASP.NETMVC差不多,但能Binding的来源更多了一

ASP.NET Core MVC的Model Binding会将HTTP Request数据,以映射的方式对应到参数中。基本上跟ASP.NET MVC差不多,但能Binding的来源更多了一些。
本篇将介绍ASP.NET Core的Model Binding。

Model Binding

要接收Client 传送来的数据,可以通过Action 的参数接收,如下:

1

2

3

4

5

6

7

8

9

10

11

12

using Microsoft.AspNetCore.Mvc;

 

namespace MyWebsite.Controllers

{

    public class HomeController : Controller

    {

        public IActionResult Index(int id)

        {

            return Content($"id: {id}");

        }

    }

}

id就是从HTTP Request的内容被Binding的Model参数。
预设的Model Binding会从HTTP Request的三个地方取值(优先顺序由上到下):

  • Form
    透过HTTP POST的form取值。如下图:

ASP.NET Core的Model Binding来源与校验的介绍

  • Route
    是通过MVC Route URL取值。
    如:http://localhost:5000/Home/Index/2id取出的值就会是2。
  • Query
    是通过URL Query参数取值。
    如:http://localhost:5000/Home/Index?id=1id取出的值就会是1。

如果三者都传入的话,会依照优先顺序取值Form > Route > Query

Binding Attributes

除了预设的三种Binding 来源外,还可以通过Model Binding Attributes 从HTTP Request 的其他数据中Binding。有以下6 种:

  • [FromHeader]
    从HTTP Header取值。
  • [FromForm]
    通过HTTP POST的form取值。
  • [FromRoute]
    是通过MVC Route URL取值。
  • [FromQuery]
    是通过URL Query参数取值。
  • [FromBody]
    从HTTP Body取值,通常用于取JSON, XML。
    ASP.NET Core MVC预设的序列化是使用JSON,如果要传XML格式做Model Binding的话,要在MVC服务加入XmlSerializerFormatters,如下:

    Startup.cs

1

2

3

4

5

6

// ...

public void ConfigureServices(IServiceCollection services)

{

    services.AddMvc()

            .AddXmlSerializerFormatters();

}

  • [FromServices]
    这个比较特别,不是从HTTP Request取值,而是从DI容器取值。
    DI预设是使用Constructor Injection,但Controller可能会因为每个Action用到不一样的Service导致很多参数,所以也可以在Action注入Service。

范例程序

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

// ...

public class HomeController : Controller

{

    public IActionResult FirstSample(

        [FromHeader]string header,

        [FromForm]string form,

        [FromRoute]string id,

        [FromQuery]string query)

    {

        return Content($"header: {header}, form: {form}, id: {id}, query: {query}");

    }

     

    public IActionResult DISample([FromServices] ILogger logger)

    {

        return Content($"logger is null: {logger == null}.");

    }

 

    public IActionResult BodySample([FromBody]UserModel model)

    {

        return Ok(model);

    }

}

 

// ...

public class UserModel

{

    public int Id { getset; }       

    public string Name { getset; }       

    public string Email { getset; }       

    public string PhoneNumber { getset; }       

    public string Address { getset; }

}

输出结果

FirstSample输出结果:

ASP.NET Core的Model Binding来源与校验的介绍

DISample输出结果:
http://localhost:5000/Home/DISample

1

logger is null: False.

BodySample输出结果:

  • JSON
    ASP.NET Core的Model Binding来源与校验的介绍
  • XML
    ASP.NET Core的Model Binding来源与校验的介绍

Model 验证

Model Binding 也可以顺便帮忙验证字段数据,只要在字段的属性上面带上Validation Attributes,如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

using System.ComponentModel.DataAnnotations;

// ...

public class UserModel

{

    [Required]

    public int Id { getset; }

 

    [RegularExpression(@"\w+")]

    [StringLength(20, MinimumLength = 4)]

    public string Name { getset; }

 

    [EmailAddress]

    public string Email { getset; }

 

    [Phone]

    public string PhoneNumber { getset; }

 

    [StringLength(200)]

    public string Address { getset; }

}

然后在Action 加上判断:

Controllers\HomeController.cs

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

using Microsoft.AspNetCore.Mvc;

 

namespace MyWebsite.Controllers

{

    public class HomeController : Controller

    {

        // ...

        public IActionResult BodySample([FromBody]UserModel model)

        {

            // 由于 Id 是 int 类型,int 默认为 0

            // 虽然带上了 [Required],但不是 null 所以算是有值。

            if (model.Id <1)

            {

                ModelState.AddModelError("Id""Id not exist");

            }

            if (ModelState.IsValid)

            {

                return Ok(model);

            }

            return BadRequest(ModelState);

        }

    }

}

输入错误数据的输出结果:

ASP.NET Core的Model Binding来源与校验的介绍

.NET Core提供了很多的Validation Attributes,可以参考官网:System.ComponentModel.DataAnnotations

自定义Validation Attributes

如果.NET Core提供的Validation Attributes不够用还可以自己做。
例如上述范例的数据模型多了生日字段,需要验证年龄:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

using System;

using System.ComponentModel.DataAnnotations;

 

namespace MyWebsite.Attributes

{

    public class AgeCheckAttribute : ValidationAttribute

    {

        public int MinimumAge { getprivate set; }

        public int MaximumAge { getprivate set; }

 

        public AgeCheckAttribute(int minimumAge, int maximumAge)

        {

            MinimumAge = minimumAge;

            MaximumAge = maximumAge;

        }

 

        protected override ValidationResult IsValid(object value, ValidationContext validationContext)

        {

            var date = Convert.ToDateTime(value);

 

            if (date.AddYears(MinimumAge) > DateTime.Today

                || date.AddYears(MaximumAge)

            {

                return new ValidationResult(GetErrorMessage(validationContext));

            }

 

            return ValidationResult.Success;

        }

 

        private string GetErrorMessage(ValidationContext validationContext)

        {

            // 有帶 ErrorMessage 的话优先使用

            // [AgeCheck(18, 120, ErrorMessage="xxx")]

            if (!string.IsNullOrEmpty(this.ErrorMessage))

            {

                return this.ErrorMessage;

            }

 

            // 自定义错误信息

            return $"{validationContext.DisplayName} can't be in future";

        }

    }

}

参考

Overview of ASP.NET Core MVC 
Introduction to model validation in ASP.NET Core MVC 
ASP.NET CORE 2.0 MVC MODEL BINDING 
ASP.NET CORE 2.0 MVC MODEL VALIDATION

 

a learning project for net core 2: https://github.com/SnailDev/SnailDev.NETCore2Learning

 


推荐阅读
  • 在springmvc框架中,前台ajax调用方法,对图片批量下载,如何弹出提示保存位置选框?Controller方法 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • Spring常用注解(绝对经典),全靠这份Java知识点PDF大全
    本文介绍了Spring常用注解和注入bean的注解,包括@Bean、@Autowired、@Inject等,同时提供了一个Java知识点PDF大全的资源链接。其中详细介绍了ColorFactoryBean的使用,以及@Autowired和@Inject的区别和用法。此外,还提到了@Required属性的配置和使用。 ... [详细]
  • 使用圣杯布局模式实现网站首页的内容布局
    本文介绍了使用圣杯布局模式实现网站首页的内容布局的方法,包括HTML部分代码和实例。同时还提供了公司新闻、最新产品、关于我们、联系我们等页面的布局示例。商品展示区包括了车里子和农家生态土鸡蛋等产品的价格信息。 ... [详细]
  • 本文介绍了lua语言中闭包的特性及其在模式匹配、日期处理、编译和模块化等方面的应用。lua中的闭包是严格遵循词法定界的第一类值,函数可以作为变量自由传递,也可以作为参数传递给其他函数。这些特性使得lua语言具有极大的灵活性,为程序开发带来了便利。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • springmvc学习笔记(十):控制器业务方法中通过注解实现封装Javabean接收表单提交的数据
    本文介绍了在springmvc学习笔记系列的第十篇中,控制器的业务方法中如何通过注解实现封装Javabean来接收表单提交的数据。同时还讨论了当有多个注册表单且字段完全相同时,如何将其交给同一个控制器处理。 ... [详细]
  • Python瓦片图下载、合并、绘图、标记的代码示例
    本文提供了Python瓦片图下载、合并、绘图、标记的代码示例,包括下载代码、多线程下载、图像处理等功能。通过参考geoserver,使用PIL、cv2、numpy、gdal、osr等库实现了瓦片图的下载、合并、绘图和标记功能。代码示例详细介绍了各个功能的实现方法,供读者参考使用。 ... [详细]
  • 本文介绍了在mac环境下使用nginx配置nodejs代理服务器的步骤,包括安装nginx、创建目录和文件、配置代理的域名和日志记录等。 ... [详细]
  • WebSocket与Socket.io的理解
    WebSocketprotocol是HTML5一种新的协议。它的最大特点就是,服务器可以主动向客户端推送信息,客户端也可以主动向服务器发送信息,是真正的双向平等对话,属于服务器推送 ... [详细]
  • 怎么在PHP项目中实现一个HTTP断点续传功能发布时间:2021-01-1916:26:06来源:亿速云阅读:96作者:Le ... [详细]
  • SpringMVC接收请求参数的方式总结
    本文总结了在SpringMVC开发中处理控制器参数的各种方式,包括处理使用@RequestParam注解的参数、MultipartFile类型参数和Simple类型参数的RequestParamMethodArgumentResolver,处理@RequestBody注解的参数的RequestResponseBodyMethodProcessor,以及PathVariableMapMethodArgumentResol等子类。 ... [详细]
  • 本文介绍了一种在PHP中对二维数组根据某个字段进行排序的方法,以年龄字段为例,按照倒序的方式进行排序,并给出了具体的代码实现。 ... [详细]
  • 如何在php文件中添加图片?
    本文详细解答了如何在php文件中添加图片的问题,包括插入图片的代码、使用PHPword在载入模板中插入图片的方法,以及使用gd库生成不同类型的图像文件的示例。同时还介绍了如何生成一个正方形文件的步骤。希望对大家有所帮助。 ... [详细]
  • JavaWeb中读取文件资源的路径问题及解决方法
    在JavaWeb开发中,读取文件资源的路径是一个常见的问题。本文介绍了使用绝对路径和相对路径两种方法来解决这个问题,并给出了相应的代码示例。同时,还讨论了使用绝对路径的优缺点,以及如何正确使用相对路径来读取文件。通过本文的学习,读者可以掌握在JavaWeb中正确找到和读取文件资源的方法。 ... [详细]
author-avatar
猫儿爱妞_591
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有