【Entity Framework】闲话EF中批量配置

news/2024/5/17 16:34:15

【Entity Framework】闲话EF中批量配置

文章目录

  • 【Entity Framework】闲话EF中批量配置
    • 一、概述
    • 二、OnModelCreating中的批量配置
      • 元数据API的缺点
    • 三、预先约定配置
      • 忽略类型
      • 默认类型映射
      • 预先约定配置的限制
      • 约定
      • 添加新约定
      • 替换现有约定
      • 约定实现注意事项
    • 四、何时使用每种方法进行批量配置
      • 在以下情况下使用`元数据API`:
      • 在以下情况下使用预先预定模型配置:
      • 在以下情况下使用最终约定:
      • 在以下情况下使用交互式约定:

在这里插入图片描述

一、概述

当需要在多个实体类型中以相同方式配置一个方面时,可以通过以下方式减少代码重复并合并逻辑。

二、OnModelCreating中的批量配置

ModelBuilder返回的每个生成器对象都会公开一个ModelMetadata属性,该属性提供对构成模型的对象的低级别访问。具体而言,有些方法允许循环访问模型中的特定对象,并对其应用通用配置。

如下示例中,模型中包含一个自定义值类型Currency:

public readonly struct Current
{public Crrent(decimal amount)=>Amount = amount;public decimal Amount{get;}public override string ToString() => $"${Amount}";
}

默认情况下不会发现此类型的属性,因为当前EF提供程序不知道如何将其映射到数据库类型。此OnModelCreating代码片段添加类型Currency的所有属性。并将值类型器配置为受支持的类型-decimal

foreach(var entityType in modelBuilder.Model.GetEntityTypes())
{foreach (var propertyInfo in entityType.ClrType.GetProperties()){if (propertyInfo.PropertyType == typeof(Currency)){entityType.AddProperty(propertyInfo).SetValueConverter(typeof(CurrencyConverter));}}
}
public class CurrencyConverter : ValueConverter<Currency, decimal>
{public CurrencyConverter(): base(v => v.Amount,v => new Currency(v)){}
}

元数据API的缺点

  • Fluent API不同,对模型的每次修改都需要显示完成。
  • 每次更改后都会运行约定。如果删除约定发现的导航,则约定将再次运行,并可以将导航添加回来。为了防止这种情况发生,需要延迟约定,直到通过调用DelayConventions() 添加属性之后再释放返回的对象,或者使用AddIgnored 将CLR属性标记为忽略。
  • 发生此循环访问后,可能会添加实体类型,并且不会向其应用配置。通常可以通过将此代码放在OnModelCreating的末尾来防止这种情况,但如果有两组相互依赖的配置,则可能没有一个顺序可以一致地应用这些配置。

三、预先约定配置

EF Core允许为给定CLR类型指定一次映射配置;然后,此配置将应用于模型中发现的给定类型的所有属性。这称为"预先预定模型配置",因为它配置模型的各个方面,直到允许运行模型生成约定。此类配置通过在DbContext派生的类型上替代ConfigureConventions来应用。

以下示例演示如何将类型Currency的所有属性配置为具有值转换器:

protected override void ConfigureConventions(ModelConfigurationBuilder configurationBuilder){configurationBuilder.Properties<Currency>().HaveConversion<CurrencyConverter>();
}

以下示例演示如何在类型 string 的所有属性上配置一些方面:

configurationBuilder.Properties<string>().AreUnicode(false).HaveMaxLength(1024);

备注

ConfigureConventions调用中指定的类型可以是基类型,接口或泛型类型定义。所有匹配的配置将从最不具体的配置开始按顺序应用:

  1. 接口
  2. 基类型
  3. 泛型类型定义
  4. 不可以为null的值类型
  5. 确切类型

预先约定配置相当于将匹配对象添加到模型后立即应用的显示配置。它将替代所有约定和数据注释。

忽略类型

预先预定配置还允许忽略某个类型,并防止约定将其作为实体类型或实体类型的属性发现:

configurationBuilder.IgnoreAny(typeof(IList<>));

默认类型映射

通常,只要为此类型的属性指定了值转换器,EF就可以使用提供程序不支持的类型常量转换查询。但是,在不涉及此类型的任何属性的查询中,EF无法找到正确的值转换器。在这种情况下。可以调用DefaultTypeMapping 添加或替代提供程序类型映射:

configurationBuilder.DefaultTypeMapping<Currency>().HasConversion<CurrencyConverter>();

预先约定配置的限制

  • 许多方面无法使用此方法进行配置
  • 目前,配置仅由CLR类型确定
  • 在创建模型之前,将执行此配置。如果应用此配置时出现任何冲突,则异常堆栈跟踪将不包含ConfigureConventions方法,因此可能更难找到原因。

约定

EF Core模型生成约定是根据在生成模型时对模型的更改触发的包含逻辑的类。这使得模型在进行显示配置,应用映射属性以及运行其他约定时保持最新状态。为了参与此过程,每个约定实现一个或多个接口,用于确定何时触发相应的方法。

模型生成约定是控制模型配置的强大方法,但可能很复杂且难以处理得当。 在许多情况下,可以使用现有的预先约定模型配置来轻松指定属性和类型的常见配置。

添加新约定

每个层次结构一个表继承映射策略需要一个鉴别器列来指定任何定行中表示的类型。默认情况下,EF对鉴别器使用未绑定的字符串列,这可确保它使用于任何鉴别器长度。但是,限制鉴别器字符串的最大长度可能会提高存储和查询的效率。

EF Core模型生成约定是根据在生成模型时对模型的更改触发,这使得模型在进行显示配置,应用映射属性以及运行其他约定时保持最新状态,为了参与此过程,每个约定实现一个或多个接口,用于确定何时触发约定。如,每次向模型添加新实体类型时,都会触发实现IEntityTypeAddedConvention的约定。同样,每当将键或外键添加到模型时,都会触发实现IForeignKeyAddedConventionIKeyAddedConvention的约定。

替换现有约定

有时,我们不想完全删除现有约定,而是想将其替换为一种操作基本相同但行为已更改的约定。这很有用,因为现有约定已经实现了需要适当触发的接口。

约定实现注意事项

EF Core会跟踪每个配置是如何进行的。这由ConfigurationSource枚举表示。不同类型的配置包含:

  • Explicit:模型元素在OnModelCreating中显示配置
  • DataAnnotation:模型元素是使用CLR类型的映射属性(既数据注释)配置的
  • Convention:模型元素是由模型生成约定配置的

预定永远不会替代标记为DataAnnotationExplicit的配置。这是通过使用"约定生成器"来实现的。例如,从Builder属性获取的IConventionPropertyBuilder

property.Builder.HasMaxLength(512);

如果 HasMaxLength 尚未由映射属性配置或在 OnModelCreating 中配置,则在约定生成器上调用它只会设置最大长度。

此类生成器方法还有第二个参数:fromDataAnnotation。 如果约定代表映射属性进行配置,则将其设置为 true

四、何时使用每种方法进行批量配置

在以下情况下使用元数据API

  • 配置需要在某个时间应用,而无需对模型中的后续更改做出反应。
  • 模型生成速度非常重要。元数据API的安全检查较少,因此比其他方法稍快一些,但是使用编译模型会产生更好的启动时间。

在以下情况下使用预先预定模型配置:

  • 适用条件很简单,因为它仅取决于类型。
  • 需要在模型中添加给定类型的属性并替代数据注释和约定时,随时应用配置。

在以下情况下使用最终约定:

  • 适用条件很复杂。
  • 配置不应替代数据注释指定的内容。

在以下情况下使用交互式约定:

  • 多个约定相互依赖。 最终约定按照添加顺序运行,因此无法对后面的最终约定所做的更改做出反应。
  • 逻辑在多个上下文之间共享。 交互式约定比其他方法更安全。

http://www.mrgr.cn/p/70788173

相关文章

Rust实现hotkey

最近想写一个快捷启动工具来满足自己的需求,刚好用自己最喜欢的 Rust 来实现实现方法 文档: global_hotkey 具体代码 use global_hotkey::{hotkey::{Code, HotKey, Modifiers},GlobalHotKeyEvent, GlobalHotKeyManager, HotKeyState, }; use winit::event_loop::{ControlFlow,…

2024面试软件测试,常见的面试题(上)

一、综合素质 1、自我介绍 面试官您好&#xff0c;我叫XXX&#xff0c;一直从事车载软件测试&#xff0c;负责最多的是中控方面。 以下是我的一些优势&#xff1a; 车载的测试流程我是熟练掌握的&#xff0c;且能够独立编写测试用例。 平时BUG提交会使用到Jira&#xff0c;类似…

海外云手机为什么适合社媒运营?

如今&#xff0c;社媒营销如果做得好&#xff0c;引流效果好的账号&#xff0c;可以用来带货变现&#xff0c;而外贸、品牌出海也同样都在做社媒营销&#xff0c;Tik Tok、facebook、ins等热门的海外社媒平台都是行业密切关注的&#xff0c;必要的时候&#xff0c;大家会使用海…

高级IO和5种IO模型

目录 1. 高级IO1.1 IO的基本概念1.2 OS如何得知外设当中有数据可读取1.3 OS如何处理从网卡中读取到的数据包1.4 IO的步骤 2. 五种IO模型2.1 利用钓鱼来理解2.2 阻塞IO2.3 非阻塞IO2.4 信号驱动IO2.5 IO多路转接2.6 异步IO 3. 高级IO的概念3.1 同步通信 VS 异步通信3.2 阻塞 VS …

FineReport11 报表技巧02- 实现类Excel表头筛选功能

背景: 在报表开发中,有的需求方用习惯Excel的表头筛选时,就不太习惯帆软的特意点击报表控件进行筛选,希望报表筛选方式可以类似Excel那种直接在表头进行筛选的功能 最终效果如下:实现步骤: 1.1、数据集准备 产品信息表: SELECT 客户,产品,数量,cast(下单时间 as date) a…

项目冲刺

项目冲刺汇总 第一天 第二天 会议图片第三天 会议图片第四天 会议图片第五天 会议图片第六天 会议图片第七天 会议图片燃尽图

正则表达式(Regular Expression)

正则表达式很重要&#xff0c;是一个合格攻城狮的必备利器&#xff0c;必须要学会&#xff01;&#xff01;&#xff01; &#xff08;参考视频&#xff09;10分钟快速掌握正则表达式&#xff08;奇乐编程学院&#xff09;https://www.bilibili.com/video/BV1da4y1p7iZ在线测试…

CommunityToolkit.Mvvm笔记---ObservableValidator

ObservableValidator 是实现 INotifyDataErrorInfo 接口的基类&#xff0c;它支持验证向其他应用程序模块公开的属性。 它也继承自 ObservableObject&#xff0c;因此它还可实现 INotifyPropertyChanged 和 INotifyPropertyChanging。 它可用作需要支持属性更改通知和属性验证的…

MyBatis 中的动态 SQL 的相关使用方法

为什么会有动态SQL&#xff0c;把SQL写死不是比较方便吗&#xff1f;其实有很多的举例&#xff0c;这里我那一个常见的来说&#xff0c;像我们用户注册&#xff0c;会有必填字段和非必填字段&#xff0c;有些传来的参数不一样&#xff0c;那对应的SQL也不一样&#xff0c;因此&…

HarmonyOS NEXT应用开发之下拉刷新与上滑加载案例

介绍 本示例介绍使用第三方库的PullToRefresh组件实现列表的下拉刷新数据和上滑加载后续数据。 效果图预览使用说明进入页面,下拉列表触发刷新数据事件,等待数据刷新完成。 上滑列表到底部,触发加载更多数据事件,等待数据加载完成。实现思路使用第三方库pullToRefresh组件,…

基于51单片机的自行车测速里程码表设计( proteus仿真+程序+设计报告+原理图+讲解视频)

基于51单片机的自行车测速里程码表设计 1. 主要功能&#xff1a;2. 讲解视频&#xff1a;3. 仿真设计4. 程序代码5. 设计报告6. 原理图7. 设计资料内容清单资料下载链接&#xff1a; 基于51单片机的自行车测速里程码表设计( proteus仿真程序设计报告原理图讲解视频&#xff09;…

Redis Cluster 集群部署

目录一、什么是redis Cluster集群二、集群架构图三、redis Cluster部署架构1、测试环境2、生产环境四、原生命令手动部署Redis Cluster1、环境准备2、为所有节点启用redis集群支持3、执行meet操作实现互相通信在任意一节点上和其它所有节点进行meet通信,以m1为例4、为每个mast…

2024-9.python文件操作

文件操作 引言 到目前为止&#xff0c;我们做的一切操作&#xff0c;都是在内存里进行的&#xff0c;这样会有什么问题吗&#xff1f;如果一旦断电或发生意外关机了&#xff0c;那么你辛勤的工作成果将瞬间消失。是不是感觉事还挺大的呢&#xff1f;现在你是否感觉你的编程技…

第七天博客

第七天随笔 会议照昨天工作完成总结 黄永名: 今天的测试人员测出了一些bug,有关于sql语句的。不过文档有点不太直观,细致的看了一下。出现了无法通过作者名检索书籍的情况,于是在数据库上用sql语句测试了一下,并没有出现问题。还有很多类似这样的情况,比如无法注册管理员…

conda安装好了虚拟环境,可以在pycham里导入xpotato库,但是在jupyter notebook中却导入不了

问题描述 使用conda安装好了虚拟环境&#xff0c;可以在pycham里导入xpotato库&#xff0c;但是在jupyter notebook中却导入不了 原因 虽然是在anaconda Prompt的环境中进入的Jupyter Notebook&#xff0c;但是Jupyter Notebook的默认内核却不是那个环境&#xff0c;需要重新更…

Vue3从入门到实践:深度了解新组件

1.Teleport 概念&#xff1a;Teleport&#xff08;传送门&#xff09;是一个新的特性&#xff0c;用于在DOM中的任意位置渲染组件。它允许你将组件的内容渲染到DOM中的另一个位置&#xff0c;而不受组件层次结构的限制。 下面举出例子解释&#xff1a; 1.新建App.vue文件作…

古典密码之棋盘密码(ADFGVX,ADFGX,Polybius)

引言 棋盘密码只加密字母,不区分大小写(ADFGVX棋盘还可加密数字),若输入其它字符则原样保留。若输入多行(段落),每行是单独加密的。 Polybius/ADFGX棋盘由于I/J在同一格,解密后需人工确认I和J。 ADFGX/ADFGVX棋盘原则上需要密钥二次加密,但可留空不使用密钥。若使用,…

射影几何学笔记

给大家拉坨大的。 在中学阶段,我们就研究过欧几里得平面上的几何。在初中阶段我们学习了平移与旋转,在高中阶段我们学习了仿射,这些几何变换有一个共同点:保持共线三点与共点三线在变换后仍共线或共点。然而在生活中,除了这些变换以外,还有更一般的变换也拥有这个性质:比…

Linux的学习之路:5、粘滞位与vim

摘要 这里主要是把上章没说完的权限的粘滞位说一下&#xff0c;然后就是vim的一些操作。 目录 摘要 一、粘滞位 二、权限总结 三、vim的基本概念 四、vim的基本操作 五、vim正常模式命令集 1、插入模式 2、从插入模式切换为命令模式 3、移动光标 4、删除文字 5、复…