前言

记入一些零零碎碎的知识。

Shadow Properties

参考:Docs – Shadow and Indexer Properties

Shadow Property 指的是那些在数据库有 Column 但是在 Entity Class 却没有 Property 的 Property。

Foreign key example

举例

public class Product
{
public int Id { get; set; }
public string Name { get; set; } = "";
public List<Color> Colors = [];
} public class Color
{
public int Id { get; set; }
public string Name { get; set; } = "";
public int ProductId { get; set; }
public Product Product { get; set; } = null!;
}

有 2 个 Entity -- Product 和 Color,它们的关系是一对多。在数据库 Color Table 里一定要记入一个 foreign key ProductId,这样它俩才能关联在一起。

在 Entity Class,关联 Product 和 Color 的是 Product.List<Color> 和 Color.Product 属性,Color.ProductId 属性是多余的。

这个多余的 ProductId 就很符合 Shadow Property 的理念 -- 数据库有 Column 但是在 Entity Class 却没有 Property。

具体做法很简单,首先把 ProductId 从 class Color 中移除

public class Color
{
public int Id { get; set; }
public string Name { get; set; } = "";
// public int ProductId { get; set; } // 不需要了
public Product Product { get; set; } = null!;
}

然后在设置 Fluent API 时 HasForeignKey 直接写 string "ProductId"。

modelBuilder.Entity<Color>()
.HasOne(e => e.Product)
.WithMany(e => e.Colors)
// .HasForeignKey(e => e.ProductId) 把 e => e.ProductId 改成 "ProductId"
.HasForeignKey("ProductId")
.OnDelete(DeleteBehavior.Cascade);

这样 ProductId 就变成 Shadow Property 了。

using var db = new ApplicationDbContext();
var colorEntity = db.Model.GetEntityTypes().Single(e => e.ClrType == typeof(Color));
var productIdProperty = colorEntity.GetProperty("ProductId");
Console.WriteLine("is shadow: " + productIdProperty.IsShadowProperty()); // is shadow: true

Config and access shadow properties

config shadow property

上面例子中,我们利用 HasForeignKey 间接设置了 Shadow Property。

我们来一个直接的。

modelBuilder.Entity<Color>().Property<DateTimeOffset>("LastUpdated");

使用 Property 方法,提供一个类型和一个 name 就可以了,后续要链接上 HasMaxLength,HasPrecision,HasColumnName 都可以。

access shadow property

LastUpdated shadow property 不在 class 里,那要怎样 set value 呢?

答案是通过 entry property 直接 set CurrentValue

using var db = new ApplicationDbContext();
var color = new Color { Name = "red" };
db.Entry(color).Property("LastUpdated").CurrentValue = DateTimeOffset.Now;
db.Colors.Add(color);
db.SaveChanges();

通常这些逻辑会写在 Entity Interception 里。

Backing Fields

参考:Docs – Backing Fields

下面是一个 Order Entity

public class Order
{
public int Id { get; set; } private string _name = "";
public string Name
{
get { return _name; }
set { _name = value; }
}
}

它的特别之处在于,它有一个 field _name,然后有 getter setter Property Name。

有时候在 Application Level 我们想使用 getter setter 做一些额外处理,但又不希望 Database Layer 受到影响。

这时就可以使用 Backing Fields,它可以让 Entity 和 数据库映射的时候不使用 Name Property,而是直接使用 _name Field。

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Entity<Order>(
builder =>
{
builder.ToTable("Order");
builder.Property(e => e.Name).HasField("_name").UsePropertyAccessMode(PropertyAccessMode.Field);
});
}

告诉它,Order 有一个 Property Name,它有一个 Field _name,访问的时候使用 Field。

query 数据后会实例化 Order,这个时候它会直接写入 _name 而不是通过 Name setter。

在 translate SQL command 的时候会直接访问 _name value,而不是 Name getter。

PropertyAccessMode 是 Enum,它有好几个选项可以分开控制 query 数据时和 translate SQL command 时要用 Field 还是 Property。

AsAsyncEnumerable

在 ASP.NET Core – ADO.NET 文章中,我们有提到

SQL 返回的 data 是 stream 来的,读的时候是一条一条异步读取的。

这种读取方式的好处是可以在数据还没有加载完之前,就开始对已加载的数据做处理。

然而 EF Core ToListAsync 是把 reader 读到完放进 List 里面以后才让我们使用。

var products = await db.Products.ToListAsync();

在 C# 8.0 之前,我们无法透过 EF Core 使用到 ADO.NET async read row by row 的好处。

直到 C# 8.0 推出 Iterating with Async Enumerables 后才可以。

using var db = new ApplicationDbContext();

var products = db.Products.AsAsyncEnumerable();

await foreach (var product in products)
{
Console.WriteLine(product.Name);
}

AsAsyncEnumerable + await foreach 就等价于 ADO.NET 的 whilte (await reader.ReadAsync)。

在 ADO.NET reader 还没有读取完毕之前,我们就可以拿到已读取的 entity 做操作了。

IAsyncEnumerable to List

有些 Library 会因为担心性能而选择 return IAsyncEnumerable,出发点是好,但是对于使用者来说,未必是件好事,因为 IAsyncEnumerable 比 List 难用多了。

于是,就有了 IAsyncEnumerable 转换成 List 的需求。

我们可以使用一个 Library 做到这个事情 -- System.Linq.Async (它是 Rx.NET 派系的一员)

dotnet add package System.Linq.Async 
var products = await db.Products.AsAsyncEnumerable().ToListAsync();
Console.WriteLine(products.Count);

它扩展了 IAsyncEnumerable 接口,让它多了 ToListAsync 方法可以用。

它的具体实现很简单,就是 for loop 放进 List 里面,源码在 ToList.cs

EF Core – 大杂烩的更多相关文章

  1. EF Core 数据库迁移(Migration)

    工具与环境介绍 1.开发环境为vs 2015 2.mysql EF Core支持采用  Pomelo.EntityFrameworkCore.MySql   源代码地址(https://github. ...

  2. Asp.net Core 通过 Ef Core 访问、管理Mysql

    本文地址:http://www.cnblogs.com/likeli/p/5910524.html 环境 dotnet Core版本:1.0.0-preview2-003131 本文分为Window环 ...

  3. EF Core 杂记

    本系列文章,将介绍本人在学习和使用EF Core的过程中的收获与心得. 或许有的地方讲的错误 欢迎大家批评指出. 1.EF Core 数据库迁移(Migration)

  4. MySQL官方.NET Core驱动已出,支持EF Core

    千呼万唤始出来MySQL官方.NET Core驱动已出,支持EF Core. 昨天MySQL官方已经发布了.NET Core 驱动,目前还是预览版,不过功能已经可用. NuGet 地址:https:/ ...

  5. EF Core 1.0 和 SQLServer 2008 分页的问题

    EF Core 1.0 在sqlserver2008分页的时候需要指定用数字分页. EF Core1.0 生成的分页语句中使用了 Featch Next.这个语句只有在SqlServer2012的时候 ...

  6. EntityFramework Core技术线路(EF7已经更名为EF Core,并于2016年6月底发布)

    官方文档英文地址:https://github.com/aspnet/EntityFramework/wiki/Roadmap 历经延期和更名,新版本的实体框架终于要和大家见面了,虽然还有点害羞.请大 ...

  7. EF Core CodeFirst实践 ( 使用MS SqlServer)

    这里使用 MS SQLSERVER ,网上大多使用 SQLite 先来一个CodeFirst 新建项目 这里我们选择  ASP.NET Core Web Application (.NET Core) ...

  8. ASP.NET Core 开发-Entity Framework (EF) Core 1.0 Database First

    ASP.NET Core 开发-Entity Framework Core 1.0 Database First,ASP.NET Core 1.0 EF Core操作数据库. Entity Frame ...

  9. ASP.NET Core 开发 - Entity Framework (EF) Core

    EF Core 1.0 Database First http://www.cnblogs.com/linezero/p/EFCoreDBFirst.html ASP.NET Core 开发 - En ...

  10. 再谈EF Core内存数据库单元测试问题

    (此文章同时发表在本人微信公众号"dotNET每日精华文章",欢迎右边二维码来关注.) 题记:在用EF Core的内存数据库进行单元测试的时候遇到"无法访问已释放的对象& ...

随机推荐

  1. WSL2连接USB设备(以USRP B210为例)

    使用WSL2时,发现其无法直接识别到宿主机上插入的USB设备. 可利用USPIPD-WIN项目进行连接. 以下以USRP B210设备连接为例,展示连接过程: 安装USBIPD-WIN 项目 参考连接 ...

  2. redis基本数据结构-有序集合

    redis基本数据结构-有序集合zset 特性 使用哈希表+跳表数据结构存储 每个元素由 分数和字段名 两部分组成 增加元素 zadd key score1 member1 [score2 membe ...

  3. 「Pygors跨平台GUI」2:安装MinGW-w64、MSYS2还是WSL2

    「Pygors系列」一句话导读: MinGW-w64只有编译器,MSYS2带着更新环境,WSL2实用性比较高 历史与渊源   Windows平台 Linux平台 二进制兼容 WSL2:运行Linux程 ...

  4. 基于 Impala 的高性能数仓建设实践之虚拟数仓

    导读:本文主要介绍网易数帆 NDH 在 Impala 上实现的虚拟数仓特性,包括资源分组.水平扩展.混合分组和分时复用等功能,可以灵活配置集群资源.均衡节点负载.提高查询并发,并充分利用节点资源. 接 ...

  5. OffscreenCanvas-离屏canvas使用说明

    OffscreenCanvas 是一个实验中的新特性,主要用于提升 Canvas 2D/3D 绘图的渲染性能和使用体验.OffscreenCanvas 的 API 很简单,但是要真正掌握好如何使用. ...

  6. 操作系统|SPOOLing(假脱机)技术

    什么是假脱机技术,它可以解决什么问题? 什么是脱机技术 要回答什么是假脱机技术,首先我们需要知道什么是脱机技术.<计算机操作系统(第四版)>写道: 为了解决人机矛盾及CPU和I/O设备之间 ...

  7. oeasy教您玩转python - 4 - # 调试程序

    ​ 调试程序 回忆上次内容 py 的程序都是写在明面上的 所有需要执行的事情都明着写到了 py 文件中 用 python3 解释 py 文件进行执行 可以下载人家写好的 py 文件 下载的 py 文件 ...

  8. mysql面试汇总

    最近一直在关注mysql方面的面试题目,并且从最近的面试情况来看,mysql在java后端的面试中,肯定是必问的题目,所以这里有必要对这块的内容进行总结,大家可以根据下面的导图进行重点复习, 引擎 1 ...

  9. 错误记录java: JDK isn't specified for module

    跑苍穹外卖的时候遇到了 java: JDK isn't specified for module 'sky-pojo'这一问题 解决办法是通过修改JDK版本,这个项目用的springboot比较早,可 ...

  10. scratch源码下载 | 蜘蛛传说

    程序说明: <蜘蛛传说>是一个通过Scratch平台制作的互动游戏项目.在这个故事中,玩家将扮演一只蜘蛛,其原本和平的生活被一只入侵的壁虎所打破.为了保卫自己的家园,蜘蛛必须运用智慧和勇气 ...