前言

简单介绍一下,值转换器和值比较器。

正文

为什么有值转换器这东西呢?

那就是这个东西一直必须存在。

比如说,我们的c# enum 对应数据库的什么呢? 是int还是string呢?

一般情况下,我们没有去写enum,保存数据库里面了一个int对吧。

那么到底谁干了这件事呢? 显然ef 是有默认的值转换器的对吧,这点是肯定的。

问题又来了,既然是有默认的值转换器,那么我的需求是不用默认的呢?

比如说:

public enum TestEnum
{
First,
Second,
Third,
Fourth
}

我想存string 怎么办呢? 为什么存string哈,姑且就说为了方便数仓清洗这一个理由,总有业务需求的嘛。

那么就需要这么做:

modelBuilder.Entity<Rider>()
.Property(e=> e.TestEnum)
.HasConversion(
v=> v.ToString(),
v => (TestEnum)Enum.Parse(typeof(TestEnum), v));

这样就可以转换成string了。

还有一个问题,那就是比如说对象映射,同一个类型的对象,会是同一中映射,比如都映射成string或者decimal,但是多个地方使用,那怎么办呢?

public class CurrencyConverter : ValueConverter<Currency, decimal>
{
public CurrencyConverter()
: base(
v => v.Amount,
v => new Currency(v))
{
}
}
protected override void ConfigureConventions(ModelConfigurationBuilder configurationBuilder)
{
configurationBuilder
.Properties<Currency>()
.HaveConversion<CurrencyConverter>();
}

嗯,还有一个问题,那就是比如说对象映射,同一个类型的对象,会是同一中映射,比如都映射成string或者decimal,但是只有一个类中多次使用呢?

如果是这样重新创建一个类,似乎不太优雅,封装性不够呀,其实我用谈一些什么封装啊啥的,就最简单一点就是不该知道的不要知道,这样大家都省心。

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);
}

这样在内部创建ValueConverter 就可以了。

那么我们来看一些内部默认的怎么写的:

var converter = new BoolToZeroOneConverter<int>();

modelBuilder
.Entity<Rider>()
.Property(e => e.TestBool)
.HasConversion(converter);

看下BoolToZeroOneConverter怎么写的哈:

就是不同类型的转换哈,很多判断,就不看了。

当然还有非常多知识,但是用不到的知识,暂时就不用去学,生活很美好,不是吗?

然后一个值比较器,这个是啥子东西呢?

我们都知道ef的一个功能就是,当我们修改我们查询的值,那么就会帮我们生成语句。

首先,最实现最傻的方式,全部更新一遍,但是显然这效率感人吧。

那么优化下,就是字段更新的,才生成更新语句。

这个思路似乎可行,那么就出现了值比较器。

modelBuilder.Entity<Rider>()
.Property(e=> e.TestList)
.HasConversion(
v=> JsonSerializer.Serialize(v, (JsonSerializerOptions)null),
v => JsonSerializer.Deserialize<List<string>>(v, (JsonSerializerOptions)null),
new ValueComparer<List<string>>(
(c1, c2)=>c1.SequenceEqual(c2),
c => c.Aggregate(0, (a, v) => HashCode.Combine(a, v.GetHashCode())),
// c tolist 的话就是复制一份
c => c.ToList()
));

为什么这里list要写这种比较呢?

那肯定是有默认的呗,默认的就是两者是否是同一个对象,如果默认情况下,引用类型赋值指向的是同一个地方,

那么你猜猜是否相同呢?

答案肯定是相同的,那么可以自定义。

ValueComparer 构造函数接受三个表达式:

  1. 用于检查相等性的表达式
  2. 用于生成哈希代码的表达式
  3. 用于截取值的快照的表达式

第一个和第二个肯定要必须存在的,比较是否相同嘛,我们知道hash和equal都是一起的,至于为什么后面equal章节会写的。

然后为什么有一个截取表达式,这个是因为其实c => c.ToList()是c进行一个快照,好为了跟后面对比,如果直接是c => c,那么无论后面增删改查都是对同一个对象比较。

大致描述了一下ef中比较重要的两个东西,下一节介绍一下ef 的关系,因为有些人刚开始学的时候不知道什么是属于,什么是拥有,这两者到底什么区别呢?

ef 值转换与值比较器的更多相关文章

  1. EF里查看/修改实体的当前值、原始值和数据库值以及重写SaveChanges方法记录实体状态

    本文目录 查看实体当前.原始和数据库值:DbEntityEntry 查看实体的某个属性值:GetValue<TValue>方法 拷贝DbPropertyValues到实体:ToObject ...

  2. javascript 中关于对象转换数字值的一些特点

    下面是摘至<Javascript 高级程序设计第三版>里的一段话 是关于对象转换数字值的一些规则 "在应用于对象时,先调用对象的valueOf()方法以取得一个可供操作的值.然后 ...

  3. ios 把毫秒值转换成日期 NSDate

    ios 把毫秒值转换成日期 (比较好用) 1343359790000 这是毫秒值------最佳解决方案-------------------- long long time=134335979000 ...

  4. Web API-如何将Controller的返回值转换成HTTP response消息

    https://www.asp.net/web-api/overview/formats-and-model-binding/json-and-xml-serialization https://co ...

  5. c++ 弧度值与角度值的转换

    Rad_to_deg --- 弧度_到_角度 的 比率Rad_to_deg = 45.0 / atan(1.0): 弧度值到角度值的转换 用 角度 = 弧度值* Rad_to_deg 角度值到弧度值的 ...

  6. 分享:根据svg节点对象类型和路径值转换坐标值

    功能用处: 对svg文件的路径节点填充时会使用(相邻两个坐标区域内的四边形的填充颜色不重复). 需要对svg文件中的Path节点或者 Polyline 节点做颜色填充.并且相邻的两个区域之间的颜色不允 ...

  7. sql 在将 nvarchar 值 转换成数据类型 int 时失败。

    假设有存储过程:proc_test2 create proc proc_test2 @Id int begin as declare @sql varchar(max) @sql = 'select ...

  8. 不再迷惑,无值和NULL值的转换

    在关系型数据库的世界中,无值和NULL值的区别是什么?一直被这个问题困扰着,甚至在写TSQL脚本时,心有戚戚焉,害怕因为自己的一知半解,挖了坑,贻害后来人,于是,本着上下求索,不达通幽不罢休的决心(开 ...

  9. json字符串转换成json对象,json对象转换成字符串,值转换成字符串,字符串转成值

    一.json相关概念 json,全称为javascript object notation,是一种轻量级的数据交互格式.采用完全独立于语言的文本格式,是一种理想的数据交换格式. 同时,json是jav ...

  10. c++ 值转换

    1.double,float 四舍五入,保留小数位数. void MainWindow::on_pushButton_clicked() { double number=3.141592; ); qD ...

随机推荐

  1. 为政务单位免费提供IP地址https证书—JoySSL

    JoySSL作为知名的证书颁发机构(CA),确实为政务单位提供了IP地址HTTPS证书的免费测试证书服务.以下是对此服务的详细介绍: 一.证书类型与特点 证书类型:JoySSL为政务单位提供了专为IP ...

  2. LLM应用实战: 给个公司简称,输出公司全称

    1.背景 本qiang~本周在处理手头项目工作的时候,遇到了一个问题,就是友方提供了一个公司名称列表(量不小~,因此无法人工处理),且该公司名称列表均为简称,需要与库中的全称做一个映射匹配. 看似简单 ...

  3. Redis未授权漏洞复现

    目录 Redis 漏洞的产生条件及利用 Redis环境搭建 漏洞复现 利用Redis写入Webshell 利用Redis写入SSH公钥 利用Redis写入计划任务 Redis安全防护 Redis re ...

  4. 数字IC知识点:处理多个时钟

    1. 多时钟域 图1.多时钟域 对于工程师来说,开发含多个时钟(见图1)的设计是一种挑战. 这样的设计中可能有以下任何一个,或者全部类型的时钟关系: 时钟的频率不同 时钟频率相同,但相位不同 以上两种 ...

  5. Redis究竟为什么这么快?

    Redis为什么这么快? 完全基于内存,数据存在内存中,绝大部分请求是纯粹的内存操作,非常快速,跟传统的磁盘文件数据存储相比,避免了通过磁盘IO读取到内存这部分的开销. 数据结构简单,对数据操作也简单 ...

  6. RMI原理及常见反序列化攻击手法

    这是对网上一些文章和视频的再总结,可以参考以下资料,师傅们分析的都挺详细了,我这就是记录一下师傅们写的博客. 廖雪峰 - 给了简单的小例子,了解即可 B站视频(白师傅) 先知社区(小阳师傅) - 讲的 ...

  7. java 中的Unsafe

    在阅读AtomicInteger的源码时,看到了这个类:sum.msic.Unsafe,之前从没见过.所以花了点时间google了一下. Unsafe的源码:http://www.docjar.com ...

  8. golang之测试testing

    01  介绍 我们使用 Golang 语言开发的项目,怎么保证逻辑正确和性能要求呢?也就是说我们如何测试我们的 Golang 代码呢?在 Golang 语言中,可以使用标准库 testing 包编写单 ...

  9. 开源的 Linux 游戏平台「GitHub 热点速览」

    <越狱>.<迷失>.<西部风云>等经典美剧背后的民间字幕翻译团队--人人影视,由于 AI 翻译的崛起.官方中文字幕的普及和版权问题,最终决定以开源的方式,为这段旅程 ...

  10. 服务迁移之《mysql数据同步问题》

    我们大概是从2022年十月份开始进行拆分的.面对一百多个服务的时候,真的是无从下手,然后公司突然空降了一个从阿里出来的架构师,然后就带着我们大刀阔斧的整体迁移. 先是服务器购买阿里云的,然后从几个核心 ...