在ASP.NET Core 中,ModelMetadataDetailsProviders 是用于配置模型元数据提供程序的核心组件,它决定了如何解析和提供模型属性的元数据(如数据类型、验证规则、显示名称等)。以下是其详细解析:

一、核心概念与作用

  1. 模型元数据(ModelMetadata)

    • 描述模型属性的元数据信息,包括:
      • 数据类型(如intstring
      • 验证特性(如[Required][MaxLength]
      • 显示名称(如[Display(Name="用户名")]
      • 绑定源(如[FromForm][FromQuery]
  2. ModelMetadataDetailsProviders 的角色

    • 提供一个扩展点,允许注册多个元数据提供程序,按顺序处理并合并元数据。
    • 支持自定义元数据逻辑,例如从数据库、配置文件或其他数据源动态生成元数据。

二、内置的元数据提供程序

ASP.NET Core 默认包含以下主要的元数据提供程序:

提供程序类型作用描述
DataAnnotationsMetadataProviderDataAnnotations特性(如[Required][Display])中提取元数据。
ValidationMetadataProvider提供数据验证相关的元数据(如[Range][RegularExpression])。
BindingMetadataProvider提供绑定相关的元数据(如[BindRequired][FromQuery])。
ConventionsMetadataProvider应用约定(Conventions)定义的元数据(如控制器 / 动作选择规则)。

三、自定义元数据提供程序

1. 实现 IModelMetadataDetailsProvider 接口
public class CustomMetadataProvider : IModelMetadataDetailsProvider
{public void CreateBindingMetadata(BindingMetadataProviderContext context){// 自定义绑定元数据(如设置默认绑定源)if (context.Key.Name == "Email"){context.BindingMetadata.BindingSource = BindingSource.Query;}}public void CreateDisplayMetadata(DisplayMetadataProviderContext context){// 自定义显示元数据(如修改显示名称)if (context.Key.Name == "UserName"){context.DisplayMetadata.DisplayName = () => "用户昵称";}}public void CreateValidationMetadata(ValidationMetadataProviderContext context){// 自定义验证元数据(如添加额外验证规则)if (context.Key.Name == "Age"){context.ValidationMetadata.ValidatorMetadata.Add(new RangeAttribute(18, 100) { ErrorMessage = "年龄必须在18-100岁之间" });}}
}
2. 注册自定义提供程序

Startup.ConfigureServices中注册:

services.AddControllersWithViews(options =>
{// 插入自定义元数据提供程序(优先级高于默认提供程序)options.ModelMetadataDetailsProviders.Add(new CustomMetadataProvider());
});

四、应用场景示例

1. 动态显示名称

从数据库或配置文件获取属性的显示名称,而非硬编码在[Display]特性中:

public class DatabaseDisplayMetadataProvider : IModelMetadataDetailsProvider
{private readonly IConfiguration _config;public DatabaseDisplayMetadataProvider(IConfiguration config){_config = config;}public void CreateDisplayMetadata(DisplayMetadataProviderContext context){// 从数据库或配置中获取显示名称var displayName = _config[$"DisplayNames:{context.Key.Name}"];if (!string.IsNullOrEmpty(displayName)){context.DisplayMetadata.DisplayName = () => displayName;}}// 其他方法实现...
}
2. 基于角色的验证规则

根据用户角色动态调整验证规则:

public class RoleBasedValidationProvider : IModelMetadataDetailsProvider
{private readonly IHttpContextAccessor _httpContextAccessor;public RoleBasedValidationProvider(IHttpContextAccessor httpContextAccessor){_httpContextAccessor = httpContextAccessor;}public void CreateValidationMetadata(ValidationMetadataProviderContext context){var user = _httpContextAccessor.HttpContext.User;// 管理员角色放宽验证if (user.IsInRole("Admin") && context.Key.Name == "Price"){context.ValidationMetadata.ValidatorMetadata.Clear(); // 移除所有验证}}// 其他方法实现...
}

五、执行顺序与优先级

  • 元数据提供程序按注册顺序执行,后注册的提供程序可以覆盖前面的元数据。
  • 示例:先应用默认提供程序,再应用自定义提供程序:
options.ModelMetadataDetailsProviders.Add(new CustomMetadataProvider()); // 后执行,优先级高
options.ModelMetadataDetailsProviders.Insert(0, new AnotherProvider());  // 先执行,优先级低

六、内置的元数据提供程序

  ExcludeBindingMetadataProvider 是一个内置的元数据提供程序,用于排除特定类型或属性的模型绑定。你提供的代码 options.ModelMetadataDetailsProviders.Add(new ExcludeBindingMetadataProvider(typeof(IModelStateService))); 的作用是阻止控制器参数绑定到 IModelStateService 类型的属性。

  1. 阻止模型绑定
    当你注册 ExcludeBindingMetadataProvider 并指定某个类型(如 IModelStateService)时,ASP.NET Core 会:

    • 忽略 HTTP 请求中与该类型相关的数据(如表单字段、查询字符串)。
    • 防止该类型的属性被模型绑定器填充值。
  2. 实现机制
    ExcludeBindingMetadataProvider 通过修改元数据中的 BindingMetadata.IsBindingAllowed 属性为 false,告诉模型绑定器跳过该类型的属性。

  3. 排除服务接口的绑定

    如果你在控制器中注入了服务(如 IModelStateService),但不想让它被请求数据绑定:
    public class MyController : Controller
    {private readonly IModelStateService _service;// 构造函数注入服务(但不希望从请求中绑定)public MyController(IModelStateService service){_service = service;}[HttpPost]public IActionResult Submit([FromForm] MyModel model){// 使用_service处理业务逻辑_service.Validate(model);return Ok();}
    }

            通过 ExcludeBindingMetadataProvider 排除 IModelStateService,可防止攻击者通过请求参数伪造服务实例。

  4. 排除敏感属性的绑定

如果你有一个包含敏感字段(如 IsAdminPasswordHash)的模型,可排除这些属性的绑定:
// 全局排除IsAdmin属性的绑定
options.ModelMetadataDetailsProviders.Add(new ExcludeBindingMetadataProvider("IsAdmin"));
这比在每个模型属性上添加 [BindNever] 特性更高效。

七、与其他组件的关系

组件与 ModelMetadataDetailsProviders 的关系
ModelBinder使用元数据确定如何绑定数据(如类型转换、绑定源)。
DataAnnotations元数据提供程序会解析DataAnnotations特性并生成对应元数据。
ModelValidator使用元数据中的验证规则执行模型验证。
Razor 视图通过元数据获取属性的显示名称、验证消息等,用于生成 HTML 标签(如asp-for)。

八、注意事项

  1. 性能考量
    避免在元数据提供程序中执行耗时操作(如数据库查询),可考虑缓存策略。

  2. 线程安全
    元数据提供程序实例通常是单例的,确保实现线程安全。

  3. 与 DataAnnotations 的兼容性
    自定义提供程序应与DataAnnotations特性协同工作,而非完全替代。

  4. 元数据缓存
    框架会缓存元数据,修改模型类后需重启应用才能生效。

总结

ModelMetadataDetailsProviders 是ASP.NET Core 中扩展模型元数据的强大机制,通过自定义元数据提供程序,你可以:

  • 动态生成显示名称、验证规则等元数据。
  • 基于运行时条件(如用户角色、配置)调整元数据。
  • 整合外部数据源(如数据库、配置文件)的元数据。

合理使用这一机制,可以实现更灵活、更动态的模型元数据管理,提升应用的可扩展性和可维护性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/914833.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/914833.shtml
英文地址,请注明出处:http://en.pswp.cn/news/914833.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

分区表设计:历史数据归档与查询加速

以下为分区表设计的核心实现方案与技术要点,综合最新技术实践整理:一、分区表核心机制与价值‌物理存储与逻辑分离‌分区表通过预定义规则(如时间戳、ID范围)将大表物理拆分为多个子表(分区),对…

下班倒计时

下班倒计时#include <stdio.h> #include <time.h> #include <unistd.h>void print_remaining_time(time_t now, time_t tar_time) {double diff difftime(tar_time, now);int hours (int)diff / 3600;int minutes ((int)diff % 3600) / 60;int seconds (…

Vue配置特性(ref、props、混入、插件与作用域样式)

前言Vue提供了许多高级特性来增强组件开发的能力。本文将深入解析Vue中的ref属性、props配置、混入(mixin)、插件开发以及scoped样式等核心特性&#xff0c;通过实例演示它们的用法&#xff0c;并给出最佳实践建议。一、ref属性详解1. ref基本用法ref用于给元素或子组件注册引用…

解析力和清晰度区别

在视觉成像、光学设备或数字信号处理领域&#xff0c;清晰度和解析力是两个相关但侧重点不同的概念。它们都与“细节呈现”有关&#xff0c;但核心定义、影响因素和应用场景存在显著区别。以下从定义、核心差异、联系三个方面详细说明&#xff1a; 一、核心定义清晰度&#xff…

Java网络通信:UDP和TCP

一、UDP特点&#xff1a; 无连接不可靠&#xff1a;通信双方不事先建立连接&#xff0c;直接发送数据。数据封装&#xff1a;将数据封装在64KB的数据包中&#xff0c;包含接收端的IP和端口。UDP通信模型&#xff1a; 模型比喻&#xff1a;以抛韭菜为例&#xff0c;发送端像抛韭…

Java行为型模式(状态模式)实现方式与测试方法

一、状态模式实现方式 核心结构 状态接口&#xff08;State&#xff09;&#xff1a;定义状态相关的行为方法。具体状态类&#xff08;ConcreteState&#xff09;&#xff1a;实现状态接口&#xff0c;封装特定状态下的逻辑。上下文类&#xff08;Context&#xff09;&#xff…

MISRA C-2012准则之标准C环境准则

目录 1.标准C环境准则 错误示例1&#xff1a;未定义行为&#xff08;整数溢出&#xff09; 错误示例2&#xff1a;未指定行为&#xff08;函数调用顺序&#xff09; 错误示例3&#xff1a;语言扩展&#xff08;GCC内置函数&#xff09; 错误示例4&#xff1a;关键未指定行…

26、鸿蒙Harmony Next开发:ArkTS并发(Promise和async/await和多线程并发TaskPool和Worker的使用)

目录 异步并发 (Promise和async/await) Promise async/await 多线程并发 多线程并发模型 内存共享模型 Actor模型 TaskPool TaskPool运作机制 TaskPool注意事项 Concurrent装饰器 装饰器说明 装饰器使用示例 TaskPool扩缩容机制 扩容机制 缩容机制 Worker Wo…

[IRF/Stack]华为/新华三交换机堆叠配置

堆叠的三大优势 提高资源利用率&#xff0c;获得更高的转发性能、链路带宽降低网络规划的复杂度、方便网络的管理降低故障对业务的影响时间 堆叠的两个需求 设备型号必须统一系统版本必须统一 华三堆叠案例&#xff1a;#### S6850_1 <H3C>sy [H3C]undo in en [H3C]sy SW…

融智兴科技: RFID超高频洗涤标签解析

在纺织品租赁与管理领域&#xff0c;布草、工服、医护织物等物品的流转追踪一直是运营管理的核心挑战。传统管理方式依赖人工计数与条码扫描&#xff0c;存在效率低下、差错率高、损耗严重等问题&#xff0c;尤其在工业洗涤环境下&#xff0c;纸质标签易损坏、识别率低。融智兴…

从平面到时空:地图故事的时空叙事与沉浸式阅读

朋友们&#xff0c;在工作中你是否也遇到过这些令人头疼的挑战&#xff1f;当项目汇报时总觉得表达不够精彩&#xff0c;方案讲解时听众总是一头雾水&#xff0c;制作应急预案时更是无从下手&#xff1f;别担心&#xff01;今天我要向大家介绍一个超级实用的解决方案——地图故…

自动控制原理知识地图:舵轮、路径与导航图

掌握自控原理的关键&#xff0c;在于看清那棵枝繁叶茂的“知识树”——从根部的数学模型&#xff0c;到主干的分析方法&#xff0c;直至顶端的系统设计。作为一名自动化专业学生&#xff0c;你是否曾在深夜里面对劳斯判据和奈奎斯特图感到深深的恐惧&#xff1f;作为初入行的工…

Flutter在Android studio运行出现Error: Entrypoint is not a Dart file

Flutter在Android studio运行出现Error: Entrypoint is not a Dart file

NE综合实验2:RIP 与 OSPF 动态路由精细配置及ACL访问控制列表 电脑

NE综合实验2&#xff1a;RIP 与 OSPF 动态路由精细配置及ACL访问控制列表 实验拓扑图实验需求 1.按照图示配置IP地址 2.按照图示区域划分配置对应的动态路由协议 3.在R7上配置dhcp服务器&#xff0c;能够让pc可以获取IP地址 4.将所有环回⼝宣告进ospf中&#xff0c;将环回⼝7宣…

Kafka 控制器(Controller)详解:架构、原理与实战

目录Kafka 控制器&#xff08;Controller&#xff09;详解&#xff1a;架构、原理与实战一、控制器的核心职责1. 元数据管理2. 分区状态机3. 故障恢复4. 集群操作协调二、传统 ZooKeeper 模式下的控制器1. 控制器选举机制2. 控制器与 ZooKeeper 的交互3. 潜在问题三、KRaft 模式…

【C++基础】#define vs constexpr:C++ 编译期常量的双雄对决(面试高频考点 + 真题解析)

​在 C++ 面试中,#define与constexpr的对比堪称 “元老级” 考点 —— 据统计,在 2023-2024 年的 C++ 工程师面试中,该知识点的出现频率高达 72%,尤其是在字节跳动、腾讯、华为等企业的校招 / 社招中,几乎是必问内容。​ 这两个语法元素都与 “编译期常量” 相关,但背后却…

k8s环境使用Operator部署Seaweedfs集群(上)

作者&#xff1a;闫乾苓 文章目录前言4.1 前置条件4.2 部署seaweedfs-operator4.3 准备operator镜像4.4 使用operator部署Seaweedfs集群4.4.1 部署StorageClass4.4.2 使用StorageClass预先创建PV前言 SeaweedFS Operator是一个Kubernetes Operator&#xff0c;用于自动化部署和…

Git CLI高危任意文件写入漏洞(CVE-2025-48384)PoC已公开

Git CLI&#xff08;命令行界面&#xff09;中存在一个高危漏洞&#xff0c;攻击者可利用该漏洞在Linux和macOS系统上实现任意文件写入。目前该漏洞的概念验证&#xff08;PoC&#xff09;利用代码已公开。该漏洞编号为CVE-2025-48384&#xff0c;CVSS严重性评分为8.1分&#x…

前端开发中关于表单内容的使用和基础知识

在前边&#xff0c;我们已经写过Web前端开发&#xff0c;Web前端开发&#xff0c;万字详细博文带你HTML&#xff0c;CSS快速入门&#xff08;上篇&#xff09;和Web前端开发&#xff0c;一文带你HTML&#xff0c;CSS快速入门&#xff08;下篇&#xff09;&#xff0c;使用近两万…

linux 安装mysql保姆教程

安装包下载地址: MySQL :: Download MySQL Community Server (Archived Versions) 解压到安装包&#xff1a; tar -zxvf mysql-5.7.36-linux-glibc2.12-x86_64.tar.gz 创建Mysql用户组和用户 groupadd mysql useradd -r -g mysql mysql 编译安装并初始化 进入/home/mysql/m…