【Entity Framework】你要知道EF中功能序列与值转换
【Entity Framework】你要知道EF中功能序列与值转换
文章目录
- 【Entity Framework】你要知道EF中功能序列与值转换
- 一、序列
- 1.1 基本用法
- 1.2 配置序列设置
- 二、值转换
- 2.1 配置值转换器
- 2.2 批量配置值转换器
- 2.3 预定义的转换
- 2.4 ValueConverter类
- 2.5 内置转换器
- 三、应用
- 3.1 简单值对象
- 3.2 复合值对象
- 3.3 基元的集合
- 3.4 值对象的集合
一、序列
序列在数据库中生成唯一的顺序数值。 序列不与特定表关联,可以设置多个表以从同一序列提取值。
1.1 基本用法
可以在模型中设置序列,然后使用它为属性生成值:
protected override void OnModelCreating(ModelBuilder modelBuilder)
{modelBuilder.HasSequence<int>("OrderNumbers");modelBuilder.Entity<Order>().Property(o => o.OrderNo).HasDefaultValueSql("NEXT VALUE FOR OrderNumbers");
}
请注意,从序列生成值的特定SQL是特定于数据库的;上面的示例适用于 SQL Server 但在其他数据库上将失败。 有关详细信息,请查阅数据库的文档。
1.2 配置序列设置
还可以配置序列的其他方面,例如其架构、起始值、增量等:
protected override void OnModelCreating(ModelBuilder modelBuilder)
{modelBuilder.HasSequence<int>("OrderNumbers", schema: "shared").StartsAt(1000).IncrementsBy(5);
}
二、值转换
值转换器可在从数据库读取或向其中写入属性值时转换属性值。此转换可以是从同一类型的一个值转换为另一个值(如:加密字符串),也可以是从一种类型的值转换为另一种类型的值(如:数据库中枚举值和字符串的相互转换)。
值转换器的指定涉及 ModelClrType 和 ProviderClrType。 模型类型是实体类型中的属性的 .NET 类型。 提供程序类型是数据库提供程序理解的 .NET 类型。 例如,若要在数据库中将枚举保存为字符串,模型类型是枚举的类型,而提供程序类型是 String。 这两种类型可以相同。
使用两个 Func 表达式树定义转换:一个从 ModelClrType 转换为 ProviderClrType,另一个从 ProviderClrType 转换为 ModelClrType。 使用表达式树的目的是使它们可被编译到数据库访问委托中,以便进行高效转换。 表达式树可能包含对复杂转换的转换方法的简单调用。
2.1 配置值转换器
值转换在DbContext.OnModelCreating中配置。如,将一个枚举和实体类型定义为:
public class Rider
{public int Id { get; set; }public EquineBeast Mount { get; set; }
}public enum EquineBeast
{Donkey,Mule,Horse,Unicorn
}
可在OnModelCreating中转换进行如下配置:在数据库中将枚举值存储为字符串。可以通过执行从ModelClrType到ProviderClrType的转换,另一个执行反向的转换:
protected override void OnModelCreating(ModelBuilder modelBuilder)
{modelBuilder.Entity<Rider>().Property(e => e.Mount).HasConversion(v => v.ToString(),v => (EquineBeast)Enum.Parse(typeof(EquineBeast), v));
}
绝不会向值转换器传递
null值。 数据库列中的 null 在实体实例中始终为 null,反之亦然。 这使实现转换更容易,并允许在可为 null 和不可为 null 的属性之间共享转换。
2.2 批量配置值转换器
为使用相关CLR类型的每个属性配置相同的值转换器很常见。可以使用预约定模型配置为整个模型执行一次此操作,而无需手动为每个属性执行此操作。要执行此操作,请将值转换器定义为类:
public class CurrencyConverter : ValueConverter<Currency, decimal>
{public CurrencyConverter(): base(v => v.Amount,v => new Currency(v)){}
}
然后按以下所示在上下文类型中重写ConfigureConventions并配置转换器:
protected override void ConfigureConventions(ModelConfigurationBuilder configurationBuilder)
{configurationBuilder.Properties<Currency>().HaveConversion<CurrencyConverter>();
}
2.3 预定义的转换
EF Core含有许多预定义转换,不需要手动编写转换函数。而是根据模型中的属性类型和请求的数据库提供程序类型选取要使用的转换。
如:下面的示例中使用了从枚举到字符串的转换,但当提供程序类型配置为string时,EF Core实际上会使用HasConversion的泛型类型自动执行此转换:
protected override void OnModelCreating(ModelBuilder modelBuilder)
{modelBuilder.Entity<Rider>().Property(e => e.Mount).HasConversion<string>();
}
可通过显式地指定数据库列类型实现相同的操作。 例如,如果实体类型的定义如下:
- 数据注释
public class Rider2
{public int Id { get; set; }[Column(TypeName = "nvarchar(24)")]public EquineBeast Mount { get; set; }
}
- Flument API
modelBuilder.Entity<Rider2>().Property(e => e.Mount).HasColumnType("nvarchar(24)");
然后,枚举值会被保存为数据库中的字符串,
2.4 ValueConverter类
上面示例调用HasConversion会创建一个ValueConverter<TModel,TProvider>实例并在属性上设置它。可改为显示地创建ValueConverter。示例如:
protected override void OnModelCreating(ModelBuilder modelBuilder)
{var converter = new ValueConverter<EquineBeast, string>(v => v.ToString(),v => (EquineBeast)Enum.Parse(typeof(EquineBeast), v));modelBuilder.Entity<Rider>().Property(e => e.Mount).HasConversion(converter);
}
多个属性使用同一个转换时,这非常有用。
2.5 内置转换器
如上所述,EF Core附带了一组预定义ValueConverter<TModel,TProvider>类,这些类位于Microsoft.EntityFrameworkCore.Storage.ValueConversion命名空间中。 在许多情况下,EF 将根据模型中属性的类型和在数据库中请求的类型,选择适当的内置转换器,正如上面的枚举转换示例所示。 例如,对 bool 属性使用 .HasConversion<int>() 会使 EF Core 将布尔值转换为数值零和一:
protected override void OnModelCreating(ModelBuilder modelBuilder)
{modelBuilder.Entity<User>().Property(e => e.IsActive).HasConversion<int>();
}
这种做法与创建一个内置BoolToZeroOneConverter<TProvider>的实例并进行显式设置在功能上的效果一样:
protected override void OnModelCreating(ModelBuilder modelBuilder)
{var converter = new BoolToZeroOneConverter<int>();modelBuilder.Entity<User>().Property(e => e.IsActive).HasConversion(converter);
}
三、应用
3.1 简单值对象
此示例使用简单类型来包装基元类型。 希望模型中的类型比基元类型更具体(因而更具类型安全性)时,这很有用。 在此示例中,该类型为 Dollars,它包装小数基元:
public readonly struct Dollars
{public Dollars(decimal amount) => Amount = amount; public decimal Amount { get; }public override string ToString() => $"${Amount}";
}
这可用于实体类型中:
public class Order
{public int Id { get; set; }public Dollars Price { get; set; }
}
还可在存储到数据库中时被转换为基本 decimal:
modelBuilder.Entity<Order>().Property(e => e.Price).HasConversion(v => v.Amount,v => new Dollars(v));
3.2 复合值对象
在上一个示例中,值对象类型仅包含一个属性。更常见的是:值对象类型组成共同构成一个域概念的多个属性。如:一个通用的Money类型包含金额和货币。
public readonly struct Money
{[JsonConstructor]public Money(decimal amount, Currency currency){Amount = amount;Currency = currency;}public override string ToString()=> (Currency == Currency.UsDollars ? "$" : "£") + Amount;public decimal Amount { get; }public Currency Currency { get; }
}
public enum Currency
{UsDollars,PoundsSterling
}
可以像以前一样在实体类型中使用此值对象:
public class Order
{public int Id { get; set; }public Money Price { get; set; }
}
值转换器目前只能执行值与一个数据库列之间的转换。此限制意味着对象的所有属性值都必须被编码为一个列值。对此,通常的处理方法是:在对象进入数据库中时序列化该对象,再在它退出数据库时反序列化。如,使用System.Text.Json:
modelBuilder.Entity<Order>().Property(e => e.Price).HasConversion(v => JsonSerializer.Serialize(v, (JsonSerializerOptions)null),v => JsonSerializer.Deserialize<Money>(v, (JsonSerializerOptions)null));
3.3 基元的集合
序列化还可用于存储基元值的集合。 如:
public class Post
{public int Id { get; set; }public string Title { get; set; }public string Contents { get; set; }public ICollection<string> Tags { get; set; }
}
再次使用System.Text.Json:
modelBuilder.Entity<Post>().Property(e => e.Tags).HasConversion(v => JsonSerializer.Serialize(v, (JsonSerializerOptions)null),v => JsonSerializer.Deserialize<List<string>>(v, (JsonSerializerOptions)null),new ValueComparer<ICollection<string>>((c1, c2) => c1.SequenceEqual(c2),c => c.Aggregate(0, (a, v) => HashCode.Combine(a, v.GetHashCode())),c => (ICollection<string>)c.ToList()));
ICollection<string> 表示可变引用类型。 也就是说,需要使用 ValueComparer,这样 EF Core 才能正确地跟踪和监测更改。
3.4 值对象的集合
结合前两个示例,我们可以创建一个值对象集合。 例如,假设有一个 AnnualFinance 类型,它为博客一年的财务状况建模:
public readonly struct AnnualFinance
{[JsonConstructor]public AnnualFinance(int year, Money income, Money expenses){Year = year;Income = income;Expenses = expenses;}public int Year { get; }public Money Income { get; }public Money Expenses { get; }public Money Revenue => new Money(Income.Amount - Expenses.Amount, Income.Currency);
}
此类型构成几个我们先前创建的 Money 类型:
public readonly struct Money
{[JsonConstructor]public Money(decimal amount, Currency currency){Amount = amount;Currency = currency;}public override string ToString()=> (Currency == Currency.UsDollars ? "$" : "£") + Amount;public decimal Amount { get; }public Currency Currency { get; }
}
public enum Currency
{UsDollars,PoundsSterling
}
然后,我们可以向实体类型添加一个 AnnualFinance 集合:
public class Blog
{public int Id { get; set; }public string Name { get; set; }public IList<AnnualFinance> Finances { get; set; }
}
接下来再次使用序列化来进行存储:
modelBuilder.Entity<Blog>().Property(e => e.Finances).HasConversion(v => JsonSerializer.Serialize(v, (JsonSerializerOptions)null),v => JsonSerializer.Deserialize<List<AnnualFinance>>(v, (JsonSerializerOptions)null),new ValueComparer<IList<AnnualFinance>>((c1, c2) => c1.SequenceEqual(c2),c => c.Aggregate(0, (a, v) => HashCode.Combine(a, v.GetHashCode())),c => (IList<AnnualFinance>)c.ToList()));
相关文章:
【Entity Framework】你要知道EF中功能序列与值转换
【Entity Framework】你要知道EF中功能序列与值转换 文章目录 【Entity Framework】你要知道EF中功能序列与值转换一、序列1.1 基本用法1.2 配置序列设置 二、值转换2.1 配置值转换器2.2 批量配置值转换器2.3 预定义的转换2.4 ValueConverter类2.5 内置转换器 三、应用3.1 简单…...
顶顶通呼叫中心中间件-SIP分机安全(mod_cti基于FreeSWITCH)
介绍 运行在公网的FreeSWITCH服务器,每天都会接收到很多恶意的呼叫请求和注册请求,尝试盗打电话。合理的配置可以防止电话给倒打,但是每天大量的攻击,会让FS产生很多日志,降低FreeSWITCH的处理能力,cti模块…...
CountDownLatch
CountDownLatch 翻译: 倒计时锁存器,,,,count计数,down停止,Latch锁 解释: 允许一个或多个线程等待,直到在其他线程中执行的一组操作完成的同步辅助不懂?…...
Vue3中的组合式API与选项式API:深入理解与比较
一、引言 Vue.js,作为前端开发的热门框架之一,以其轻量级、易上手、灵活性强等特点深受开发者的喜爱。随着Vue3的发布,其引入了全新的组合式API(Composition API),这为Vue.js的开发方式带来了新的变革。本…...
接口自动化测试实战之接口概念、项目简介及测试流程问答!
一、前言 这篇文章呢主要是想讲实战方面的内容,本文主要会讲解接口测试中的一些接口概念,流程等方面的问答,同时还会介绍一下即将要进行测试的项目,这里呢我就不多说废话了,直接进入主题吧。 二、接口概念 接口测试&…...
浏览器工作原理与实践--跨站脚本攻击(XSS):为什么Cookie中有HttpOnly属性
通过上篇文章的介绍,我们知道了同源策略可以隔离各个站点之间的DOM交互、页面数据和网络通信,虽然严格的同源策略会带来更多的安全,但是也束缚了Web。这就需要在安全和自由之间找到一个平衡点,所以我们默认页面中可以引用任意第三…...
Ubuntu配置VScode的C++环境
在Ubuntu系统下配置C环境,并运行helloworld 1. 下载VScode 我这里使用的是星火应用商店,在商店里面可以直接下载安装 http://spark-app.store/ 2.创建文件夹 3.启动VScode并打开该文件夹 4.安装以下几个扩展 PS:Clang这个插件别安装&…...
使用Code开发Django_模版和CSS
转到定义 和 查看定义 在使用Django或任何其他库的过程中,我们可能需要检查这些库中的代码。VS Code提供了两个方便的命令,可以直接导航到任何代码中的类和其他对象的定义: 转到定义 在Python开发环境中,我们可以轻松地对函数、类或者其他导入模块中的成员使用“Go to Def…...
Llama 3下月正式发布,继续开源!
4月10日,Techcrunch消息,Meta在本周伦敦举办的一场活动中确定,下个月将正式发布Llama 3并且继续开源。 Meta全球事务总裁Nick Clegg表示,我们希望在下个月,甚至更短的时间内,正式推出新一代基础模型Llama …...
有图片转成PDF文件格式的方法吗?分享图片转成PDF文件的方法
将图片转换为PDF文件是一个相对简单的过程,但也需要一定的步骤和注意事项。下面,我将详细介绍如何将图片转换为PDF文件,包括所需的工具、步骤以及可能遇到的问题和解决方案。 首先,我们需要一个能够将图片转换为PDF文件的工具。市…...
数据结构---绪论
一、绪论: 1.什么是数据? 数据是信息的载体,是描述客观事物属性的数,字符及所有能输入到计算机中并被计算机程序识别和处理的符号的集合。数据是计算机程序加工的原料。 数据元素--描述一个个体 数据元素,数据项&am…...
matlab 安装 mingw64(6.3.0),OPENEXR
matlab安装openexr 1. matlab版本与对应的mingw版本选择2. mingw(6.3.0)下载地址:3. matlab2020a配置mingw(6.3.0)流程“4. matlab 安装openexr方法一:更新matlab版本方法二:其他博文方法方法三…...
最新彩虹知识付费商城源码 V3.4
最新彩虹知识付费商城源码 V3.4,支持二级分类,多级分销,秒杀,砍价,团购,首页继续浏览,分站个人虚拟余额自定义,最新批量对接,批量下载图片,批量替换标题&…...
Redis实现延迟任务的几种方案
🏷️个人主页:牵着猫散步的鼠鼠 🏷️系列专栏:Java全栈-专栏 🏷️个人学习笔记,若有缺误,欢迎评论区指正 目录 1.前言 2.Redis如何实现延迟任务? 3.代码实现 3.1. 过期键通知事…...
一种springboot请求参数校验的实现方案
一、前提引入 很多时候,springboot提供的Restful-api需要根据业务需要进行参数校验,相应的,基于各位码友的习惯,各有各的实现方式,可谓是八仙过海各显神通。 二、常见方案 2.1 一种最原始的方法 通过if语句,对特定参数进行校验 if(null == name){return "name …...
盒子模型+响应式布局 + 原型链与继承
盒子模型 是什么 css布局基础,规定了元素在页面上如何呈现,以及元素之间的空间关系 由content paddingbordermargin四部分组成 为什么 盒子模型分为 标准盒子模型: 元素的宽度与高度 只包括content IE盒子模型: 元素的宽度与高度 包括content,padding,border 在实际操作中…...
面试准备 集合 List
ArrayList 底层实现 使用Object[] 动态数组进行存储 特性 支持存储null值非线程安全支持快速访问 初始化方法 无参–返回一个空的列表(DEFAULTCAPACITY_EMPTY_ELEMENTDATA)指定初始容量: new ArrayList(20);指定集合 new ArrayList(col…...
Java快速入门系列-7(测试与调试)
第七章:测试与调试 第7章:测试与调试7.1 单元测试(JUnit)7.1.1 为什么要进行单元测试7.1.2 JUnit基础7.1.3 断言7.1.4 测试套件7.2 集成测试与系统测试7.2.1 集成测试7.2.2 系统测试7.3 调试技巧与工具7.3.1 断点7.3.2 单步执行7.3.3 变量检查7.3.4 条件断点7.3.5 日志记录…...
算法:双指针
算法:双指针 双指针快慢指针对撞指针总结 双指针 LeetCode 283.移动零 以上题目要求我们把所有0移动到数组的末尾,也就是说,我们要把数组转化为以下状态: [ 非0区域 ] [ 0区域 ] 像这种把一个数组划分为多个区域的题型࿰…...
MySQL一些特殊功能的索引(6/16)
特殊功能性索引 B-Tree索引: InnoDB的默认索引类型,适用于多种查询操作。 可以用于等值查询、范围查询和索引列的组合查询。 创建B-Tree索引的示例: CREATE INDEX index_name ON table_name (column1, column2);全文索引(FULLTEX…...
KubeSphere 容器平台高可用:环境搭建与可视化操作指南
Linux_k8s篇 欢迎来到Linux的世界,看笔记好好学多敲多打,每个人都是大神! 题目:KubeSphere 容器平台高可用:环境搭建与可视化操作指南 版本号: 1.0,0 作者: 老王要学习 日期: 2025.06.05 适用环境: Ubuntu22 文档说…...
零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?
一、核心优势:专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发,是一款收费低廉但功能全面的Windows NAS工具,主打“无学习成本部署” 。与其他NAS软件相比,其优势在于: 无需硬件改造:将任意W…...
Oracle查询表空间大小
1 查询数据库中所有的表空间以及表空间所占空间的大小 SELECTtablespace_name,sum( bytes ) / 1024 / 1024 FROMdba_data_files GROUP BYtablespace_name; 2 Oracle查询表空间大小及每个表所占空间的大小 SELECTtablespace_name,file_id,file_name,round( bytes / ( 1024 …...
中南大学无人机智能体的全面评估!BEDI:用于评估无人机上具身智能体的综合性基准测试
作者:Mingning Guo, Mengwei Wu, Jiarun He, Shaoxian Li, Haifeng Li, Chao Tao单位:中南大学地球科学与信息物理学院论文标题:BEDI: A Comprehensive Benchmark for Evaluating Embodied Agents on UAVs论文链接:https://arxiv.…...
LeetCode - 394. 字符串解码
题目 394. 字符串解码 - 力扣(LeetCode) 思路 使用两个栈:一个存储重复次数,一个存储字符串 遍历输入字符串: 数字处理:遇到数字时,累积计算重复次数左括号处理:保存当前状态&a…...
【大模型RAG】Docker 一键部署 Milvus 完整攻略
本文概要 Milvus 2.5 Stand-alone 版可通过 Docker 在几分钟内完成安装;只需暴露 19530(gRPC)与 9091(HTTP/WebUI)两个端口,即可让本地电脑通过 PyMilvus 或浏览器访问远程 Linux 服务器上的 Milvus。下面…...
Module Federation 和 Native Federation 的比较
前言 Module Federation 是 Webpack 5 引入的微前端架构方案,允许不同独立构建的应用在运行时动态共享模块。 Native Federation 是 Angular 官方基于 Module Federation 理念实现的专为 Angular 优化的微前端方案。 概念解析 Module Federation (模块联邦) Modul…...
C++.OpenGL (10/64)基础光照(Basic Lighting)
基础光照(Basic Lighting) 冯氏光照模型(Phong Lighting Model) #mermaid-svg-GLdskXwWINxNGHso {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-GLdskXwWINxNGHso .error-icon{fill:#552222;}#mermaid-svg-GLd…...
相机Camera日志分析之三十一:高通Camx HAL十种流程基础分析关键字汇总(后续持续更新中)
【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了:有对最普通的场景进行各个日志注释讲解,但相机场景太多,日志差异也巨大。后面将展示各种场景下的日志。 通过notepad++打开场景下的日志,通过下列分类关键字搜索,即可清晰的分析不同场景的相机运行流程差异…...
3403. 从盒子中找出字典序最大的字符串 I
3403. 从盒子中找出字典序最大的字符串 I 题目链接:3403. 从盒子中找出字典序最大的字符串 I 代码如下: class Solution { public:string answerString(string word, int numFriends) {if (numFriends 1) {return word;}string res;for (int i 0;i &…...
