【Entity Framework】你要知道EF中功能序列与值转换

2024-04-16 00:12

本文主要是介绍【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);
}

二、值转换

值转换器可在从数据库读取或向其中写入属性值时转换属性值。此转换可以是从同一类型的一个值转换为另一个值(如:加密字符串),也可以是从一种类型的值转换为另一种类型的值(如:数据库中枚举值和字符串的相互转换)。

值转换器的指定涉及 ModelClrTypeProviderClrType。 模型类型是实体类型中的属性的 .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中转换进行如下配置:在数据库中将枚举值存储为字符串。可以通过执行从ModelClrTypeProviderClrType的转换,另一个执行反向的转换:

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中功能序列与值转换的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/907308

相关文章

详解如何通过Python批量转换图片为PDF

《详解如何通过Python批量转换图片为PDF》:本文主要介绍如何基于Python+Tkinter开发的图片批量转PDF工具,可以支持批量添加图片,拖拽等操作,感兴趣的小伙伴可以参考一下... 目录1. 概述2. 功能亮点2.1 主要功能2.2 界面设计3. 使用指南3.1 运行环境3.2 使用步骤4. 核

Mybatis 传参与排序模糊查询功能实现

《Mybatis传参与排序模糊查询功能实现》:本文主要介绍Mybatis传参与排序模糊查询功能实现,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、#{ }和${ }传参的区别二、排序三、like查询四、数据库连接池五、mysql 开发企业规范一、#{ }和${ }传参的

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

SpringKafka消息发布之KafkaTemplate与事务支持功能

《SpringKafka消息发布之KafkaTemplate与事务支持功能》通过本文介绍的基本用法、序列化选项、事务支持、错误处理和性能优化技术,开发者可以构建高效可靠的Kafka消息发布系统,事务支... 目录引言一、KafkaTemplate基础二、消息序列化三、事务支持机制四、错误处理与重试五、性能优

SpringIntegration消息路由之Router的条件路由与过滤功能

《SpringIntegration消息路由之Router的条件路由与过滤功能》本文详细介绍了Router的基础概念、条件路由实现、基于消息头的路由、动态路由与路由表、消息过滤与选择性路由以及错误处理... 目录引言一、Router基础概念二、条件路由实现三、基于消息头的路由四、动态路由与路由表五、消息过滤

Spring Boot 3.4.3 基于 Spring WebFlux 实现 SSE 功能(代码示例)

《SpringBoot3.4.3基于SpringWebFlux实现SSE功能(代码示例)》SpringBoot3.4.3结合SpringWebFlux实现SSE功能,为实时数据推送提供... 目录1. SSE 简介1.1 什么是 SSE?1.2 SSE 的优点1.3 适用场景2. Spring WebFlu

基于SpringBoot实现文件秒传功能

《基于SpringBoot实现文件秒传功能》在开发Web应用时,文件上传是一个常见需求,然而,当用户需要上传大文件或相同文件多次时,会造成带宽浪费和服务器存储冗余,此时可以使用文件秒传技术通过识别重复... 目录前言文件秒传原理代码实现1. 创建项目基础结构2. 创建上传存储代码3. 创建Result类4.

Python+PyQt5实现多屏幕协同播放功能

《Python+PyQt5实现多屏幕协同播放功能》在现代会议展示、数字广告、展览展示等场景中,多屏幕协同播放已成为刚需,下面我们就来看看如何利用Python和PyQt5开发一套功能强大的跨屏播控系统吧... 目录一、项目概述:突破传统播放限制二、核心技术解析2.1 多屏管理机制2.2 播放引擎设计2.3 专

一文详解SpringBoot响应压缩功能的配置与优化

《一文详解SpringBoot响应压缩功能的配置与优化》SpringBoot的响应压缩功能基于智能协商机制,需同时满足很多条件,本文主要为大家详细介绍了SpringBoot响应压缩功能的配置与优化,需... 目录一、核心工作机制1.1 自动协商触发条件1.2 压缩处理流程二、配置方案详解2.1 基础YAML