【EF Core】FromExpression 方法有什么用?
<p>比 90% 的人细心的大伙伴一定发现了 DbContext 类有一个方法叫 FromExpression,它到底干吗用的?官方文档中没有专门的介绍(只在表值函数映射的例子中看到)。</p><p>咱们先来看看此方法的签名:</p>
<div class="cnblogs_code">
<pre>IQueryable<TResult> FromExpression<TResult>(Expression<Func<IQueryable<TResult>>> expression)</pre>
</div>
<p>看着好像很复杂的样子。其实不,咱们来拆解一下:</p>
<p>1、TResult 是类型参数(泛型的知识点没忘吧),这里其实指的就是实体类型,比如,你的爱狗 Dog。</p>
<p>2、这个方法返回 IQueryable<T> 类型,说明允许你使用 LINQ 查询。</p>
<p>3、重点理解其参数——Expression<TDelegate> 表达有个万能规律:可以把与 TDelegate 类型兼容的 lambda 表达式直接赋值给 Expression<> 变量。即这个 FromExpression 方法可以使用以下 lambda 表达式作为参数:</p>
<div class="cnblogs_code">
<pre>() => [返回 IQueryable<T>]</pre>
</div>
<p>这个委托的意思就是:它,单身狗(无参数)一枚,但可以生产 IQueryable<T> 对象。</p>
<p>哦,说了一大堆,还没说这个方法到底有啥毛用。它的用处就是你可以指定一个表达式,让 EF 一开始就返回筛选过的查询。在 DbContext 的派生类中声明 DbSet<T> 类型的带 get + set 的公共属性这种生成首个查询(根查询)是最常用的方案,这个相信大伙们都很熟了,EF Core 最基础操作,老周就不多介绍了。假如你要对查询的初始数据做筛选,那么,按照 DbSet 的方案,要先执行一下 SELECT **** FROM #$$#*& 语句,然后再执行 SELECT **** FROM &*^$ WHERE xxxxx,我还没操作数据呢就执行了两次 SELECT 语句了。所以说,如果你一开始并不打算提取所有数据,那么直接从一开始就执行 SELECT **** FROM xxxx WHERE yyyy 多好,何必多浪费一条 SQL 语句?</p>
<p>还有一种使用场景:你的数据不是从某个表 SELECT 出来的,而是从一个表值函数返回的,这种情况也要借助 FromExpression 方法。</p>
<p>不知道老周以上说明你是否明白?不明白没关系,咱们实战一下你就懂了。</p>
<p>咱们先定义的实体:</p>
<div class="cnblogs_code">
<pre><span style="color: rgba(128, 128, 128, 1)">///</span> <span style="color: rgba(128, 128, 128, 1)"><summary></span>
<span style="color: rgba(128, 128, 128, 1)">///</span><span style="color: rgba(0, 128, 0, 1)"> 妖书实体
</span><span style="color: rgba(128, 128, 128, 1)">///</span> <span style="color: rgba(128, 128, 128, 1)"></summary></span>
<span style="color: rgba(0, 0, 255, 1)">public</span> <span style="color: rgba(0, 0, 255, 1)">class</span><span style="color: rgba(0, 0, 0, 1)"> Book
{
</span><span style="color: rgba(128, 128, 128, 1)">///</span> <span style="color: rgba(128, 128, 128, 1)"><summary></span>
<span style="color: rgba(128, 128, 128, 1)">///</span><span style="color: rgba(0, 128, 0, 1)"> 标识 + 主键
</span><span style="color: rgba(128, 128, 128, 1)">///</span> <span style="color: rgba(128, 128, 128, 1)"></summary></span>
<span style="color: rgba(0, 0, 255, 1)">public</span> Guid BookId { <span style="color: rgba(0, 0, 255, 1)">get</span>; <span style="color: rgba(0, 0, 255, 1)">set</span><span style="color: rgba(0, 0, 0, 1)">; }
</span><span style="color: rgba(128, 128, 128, 1)">///</span> <span style="color: rgba(128, 128, 128, 1)"><summary></span>
<span style="color: rgba(128, 128, 128, 1)">///</span><span style="color: rgba(0, 128, 0, 1)"> 书名
</span><span style="color: rgba(128, 128, 128, 1)">///</span> <span style="color: rgba(128, 128, 128, 1)"></summary></span>
<span style="color: rgba(0, 0, 255, 1)">public</span> <span style="color: rgba(0, 0, 255, 1)">string</span> Title { <span style="color: rgba(0, 0, 255, 1)">get</span>; <span style="color: rgba(0, 0, 255, 1)">set</span>; } = <span style="color: rgba(0, 0, 255, 1)">string</span><span style="color: rgba(0, 0, 0, 1)">.Empty;
</span><span style="color: rgba(128, 128, 128, 1)">///</span> <span style="color: rgba(128, 128, 128, 1)"><summary></span>
<span style="color: rgba(128, 128, 128, 1)">///</span><span style="color: rgba(0, 128, 0, 1)"> 简介
</span><span style="color: rgba(128, 128, 128, 1)">///</span> <span style="color: rgba(128, 128, 128, 1)"></summary></span>
<span style="color: rgba(0, 0, 255, 1)">public</span> <span style="color: rgba(0, 0, 255, 1)">string</span>? Description { <span style="color: rgba(0, 0, 255, 1)">get</span>; <span style="color: rgba(0, 0, 255, 1)">set</span><span style="color: rgba(0, 0, 0, 1)">; }
</span><span style="color: rgba(128, 128, 128, 1)">///</span> <span style="color: rgba(128, 128, 128, 1)"><summary></span>
<span style="color: rgba(128, 128, 128, 1)">///</span><span style="color: rgba(0, 128, 0, 1)"> 作者
</span><span style="color: rgba(128, 128, 128, 1)">///</span> <span style="color: rgba(128, 128, 128, 1)"></summary></span>
<span style="color: rgba(0, 0, 255, 1)">public</span> <span style="color: rgba(0, 0, 255, 1)">string</span> Author { <span style="color: rgba(0, 0, 255, 1)">get</span>; <span style="color: rgba(0, 0, 255, 1)">set</span>; } = <span style="color: rgba(0, 0, 255, 1)">string</span><span style="color: rgba(0, 0, 0, 1)">.Empty;
}</span></pre>
</div>
<p>然后,继承 DbContext 类,常规操作。</p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 255, 1)">public</span> <span style="color: rgba(0, 0, 255, 1)">class</span><span style="color: rgba(0, 0, 0, 1)"> MyDbContext : DbContext
{
</span><span style="color: rgba(0, 0, 255, 1)">protected</span> <span style="color: rgba(0, 0, 255, 1)">override</span> <span style="color: rgba(0, 0, 255, 1)">void</span><span style="color: rgba(0, 0, 0, 1)"> OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
optionsBuilder.UseSqlServer(</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">Server=.\\TEST;Database=mydb;Integrated Security=True;Trust Server Certificate=True</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(0, 0, 0, 1)">);
</span><span style="color: rgba(0, 128, 0, 1)">//</span><span style="color: rgba(0, 128, 0, 1)"> 配置日志
</span><span style="color: rgba(0, 128, 0, 1)">//</span><span style="color: rgba(0, 128, 0, 1)">.LogTo(log => Debug.WriteLine(log));</span>
<span style="color: rgba(0, 0, 0, 1)"> }
</span><span style="color: rgba(0, 0, 255, 1)">protected</span> <span style="color: rgba(0, 0, 255, 1)">override</span> <span style="color: rgba(0, 0, 255, 1)">void</span><span style="color: rgba(0, 0, 0, 1)"> OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Entity</span><Book>(t =><span style="color: rgba(0, 0, 0, 1)">
{
<span style="background-color: rgba(255, 255, 153, 1)">t.ToTable(</span></span><span style="background-color: rgba(255, 255, 153, 1)"><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">books</span><span style="color: rgba(128, 0, 0, 1)">"</span>, tb =></span><span style="color: rgba(0, 0, 0, 1)">
{
tb.<span style="background-color: rgba(255, 255, 153, 1)">Property(x </span></span><span style="background-color: rgba(255, 255, 153, 1)">=> x.BookId).HasColumnName(<span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">book_id</span><span style="color: rgba(128, 0, 0, 1)">"</span></span><span style="color: rgba(0, 0, 0, 1)"><span style="background-color: rgba(255, 255, 153, 1)">)</span>;
tb.<span style="background-color: rgba(255, 255, 153, 1)">Property(z</span></span><span style="background-color: rgba(255, 255, 153, 1)">=>z.Title).HasColumnName(<span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">title</span><span style="color: rgba(128, 0, 0, 1)">"</span></span><span style="color: rgba(0, 0, 0, 1)"><span style="background-color: rgba(255, 255, 153, 1)">)</span>;
tb.<span style="background-color: rgba(255, 255, 153, 1)">Property(k </span></span><span style="background-color: rgba(255, 255, 153, 1)">=> k.Description).HasColumnName(<span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">desc</span><span style="color: rgba(128, 0, 0, 1)">"</span></span><span style="color: rgba(0, 0, 0, 1)"><span style="background-color: rgba(255, 255, 153, 1)">)</span>;
tb.<span style="background-color: rgba(255, 255, 153, 1)">Property(f </span></span><span style="background-color: rgba(255, 255, 153, 1)">=> f.Author).HasColumnName(<span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">author</span><span style="color: rgba(128, 0, 0, 1)">"</span></span><span style="color: rgba(0, 0, 0, 1)"><span style="background-color: rgba(255, 255, 153, 1)">)</span>;
})
.HasKey<span style="background-color: rgba(255, 255, 153, 1)">(t </span></span><span style="background-color: rgba(255, 255, 153, 1)">=></span><span style="color: rgba(0, 0, 0, 1)"><span style="background-color: rgba(255, 255, 153, 1)"> t.BookId)</span>;
});
}
</span><span style="color: rgba(0, 128, 0, 1)">//</span><span style="color: rgba(0, 128, 0, 1)"> 以下行现在不需要了
</span><span style="color: rgba(0, 128, 0, 1)">//</span><span style="color: rgba(0, 128, 0, 1)">public DbSet<Book> Books { get; set; }</span>
<span style="color: rgba(0, 0, 255, 1)">public</span> IQueryable<Book><span style="color: rgba(0, 0, 0, 1)"> MyBooks
</span>=> FromExpression(() => Set<Book>().Where(x => x.Author == <span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">老周</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(0, 0, 0, 1)">));
}</span></pre>
</div>
<p>这里不用再定义 DbSet<> 类型的属性了,因为我们要对数据进行筛选,重点看 MyBooks 属性的实现:</p>
<div class="cnblogs_code">
<pre> <span style="color: rgba(0, 0, 255, 1)">public</span> IQueryable<Book><span style="color: rgba(0, 0, 0, 1)"> MyBooks
</span>=> FromExpression(() => Set<Book>().<span style="background-color: rgba(255, 255, 0, 1)">Where(x => x.Author == <span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">老周</span><span style="color: rgba(128, 0, 0, 1)">"</span>)</span>);</pre>
</div>
<p>Set<Book>() 方法的调用会让 DbContext 自动在缓存字典中添加数据集合,然后一句 Where 做出筛选,上述代码的意思是只查询老周写的妖书,其他作者的不考虑。这时候 DbContext 不会发出 select * from xxx SQL 语句,所以你不用担心执行多余的 SQL。调用 FromExpression 方法后,会使初始查询直接生成 Select * from xxx where ...... 语句,只查询一次。</p>
<p>现在往 SQL Server 中新建 mydb 数据库,并创建 books 表。</p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 255, 1)">CREATE</span> <span style="color: rgba(0, 0, 255, 1)">TABLE</span> <span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">dbo</span><span style="color: rgba(255, 0, 0, 1)">]</span>.<span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">books</span><span style="color: rgba(255, 0, 0, 1)">]</span><span style="color: rgba(0, 0, 0, 1)"> (
</span><span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">book_id</span><span style="color: rgba(255, 0, 0, 1)">]</span> <span style="color: rgba(0, 0, 255, 1)">UNIQUEIDENTIFIER</span> <span style="color: rgba(0, 0, 255, 1)">DEFAULT</span> (<span style="color: rgba(255, 0, 255, 1)">newid</span>()) <span style="color: rgba(128, 128, 128, 1)">NOT</span> <span style="color: rgba(0, 0, 255, 1)">NULL</span><span style="color: rgba(0, 0, 0, 1)">,
</span><span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">title</span><span style="color: rgba(255, 0, 0, 1)">]</span> <span style="color: rgba(0, 0, 255, 1)">NVARCHAR</span> (<span style="color: rgba(128, 0, 0, 1); font-weight: bold">35</span>) <span style="color: rgba(128, 128, 128, 1)">NOT</span> <span style="color: rgba(0, 0, 255, 1)">NULL</span><span style="color: rgba(0, 0, 0, 1)">,
</span><span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">desc</span><span style="color: rgba(255, 0, 0, 1)">]</span> <span style="color: rgba(0, 0, 255, 1)">NVARCHAR</span> (<span style="color: rgba(128, 0, 0, 1); font-weight: bold">100</span>) <span style="color: rgba(0, 0, 255, 1)">NULL</span><span style="color: rgba(0, 0, 0, 1)">,
</span><span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">author</span><span style="color: rgba(255, 0, 0, 1)">]</span><span style="color: rgba(0, 0, 255, 1)">NVARCHAR</span> (<span style="color: rgba(128, 0, 0, 1); font-weight: bold">20</span>) <span style="color: rgba(128, 128, 128, 1)">NOT</span> <span style="color: rgba(0, 0, 255, 1)">NULL</span><span style="color: rgba(0, 0, 0, 1)">,
</span><span style="color: rgba(0, 0, 255, 1)">PRIMARY</span> <span style="color: rgba(0, 0, 255, 1)">KEY</span> <span style="color: rgba(0, 0, 255, 1)">CLUSTERED</span> (<span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">book_id</span><span style="color: rgba(255, 0, 0, 1)">]</span> <span style="color: rgba(0, 0, 255, 1)">ASC</span><span style="color: rgba(0, 0, 0, 1)">)
);</span></pre>
</div>
<p>顺便向表中插入些测试数据。</p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 255, 1)">insert</span> <span style="color: rgba(0, 0, 255, 1)">into</span> books (title, author, <span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">desc</span><span style="color: rgba(255, 0, 0, 1)">]</span><span style="color: rgba(0, 0, 0, 1)">)
</span><span style="color: rgba(0, 0, 255, 1)">values</span>(N<span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">R语言从盗墓到考古</span><span style="color: rgba(255, 0, 0, 1)">'</span>,N<span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">张法师</span><span style="color: rgba(255, 0, 0, 1)">'</span>, N<span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">一套不正经的R语言退隐教程</span><span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(0, 0, 0, 1)">),
(N</span><span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">疯言疯语之HTML 6.0</span><span style="color: rgba(255, 0, 0, 1)">'</span>, N<span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">老周</span><span style="color: rgba(255, 0, 0, 1)">'</span>, N<span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">提前一个世纪出现的超文本协议</span><span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(0, 0, 0, 1)">),
(N</span><span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">程序员风水学</span><span style="color: rgba(255, 0, 0, 1)">'</span>, N<span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">孙电鹰</span><span style="color: rgba(255, 0, 0, 1)">'</span>, N<span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">先匪后将的他,曾有“东陵大盗”之称,在盗掘过程中他学会了用风水理论去Debug项目</span><span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(0, 0, 0, 1)">),
(N</span><span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">鸡形机器人编程入门</span><span style="color: rgba(255, 0, 0, 1)">'</span>, N<span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">老周</span><span style="color: rgba(255, 0, 0, 1)">'</span>, N<span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">未来,由于长期不种植作物,人类只能躺在病床上依靠吮吸预制营养液维持生命;后有人提出开发鸡形机器人,帮助人类进食</span><span style="color: rgba(255, 0, 0, 1)">'</span>)</pre>
</div>
<p> </p>
<p>现在,咱们试一下。</p>
<div class="cnblogs_code">
<pre> <span style="color: rgba(0, 0, 255, 1)">using</span> <span style="color: rgba(0, 0, 255, 1)">var</span> context = <span style="color: rgba(0, 0, 255, 1)">new</span><span style="color: rgba(0, 0, 0, 1)"> MyDbContext();
</span><span style="color: rgba(0, 0, 255, 1)">foreach</span>(Book bk <span style="color: rgba(0, 0, 255, 1)">in</span><span style="color: rgba(0, 0, 0, 1)"><strong><span style="background-color: rgba(255, 255, 0, 1)"> context.MyBooks</span></strong>)
{
Console.WriteLine($</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">{bk.Author,-10}{bk.Title}</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(0, 0, 0, 1)">);
}</span></pre>
</div>
<p>如果日志启用,那么,你会看到,DbContext 从初始化到 foreach 循环访问数据,只生成了一条 SQL 语句。</p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 255, 1)">SELECT</span> <span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">b</span><span style="color: rgba(255, 0, 0, 1)">]</span>.<span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">book_id</span><span style="color: rgba(255, 0, 0, 1)">]</span>, <span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">b</span><span style="color: rgba(255, 0, 0, 1)">]</span>.<span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">author</span><span style="color: rgba(255, 0, 0, 1)">]</span>, <span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">b</span><span style="color: rgba(255, 0, 0, 1)">]</span>.<span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">desc</span><span style="color: rgba(255, 0, 0, 1)">]</span>, <span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">b</span><span style="color: rgba(255, 0, 0, 1)">]</span>.<span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">title</span><span style="color: rgba(255, 0, 0, 1)">]</span>
<span style="color: rgba(0, 0, 255, 1)">FROM</span> <span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">books</span><span style="color: rgba(255, 0, 0, 1)">]</span> <span style="color: rgba(0, 0, 255, 1)">AS</span> <span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">b</span><span style="color: rgba(255, 0, 0, 1)">]</span>
<strong><span style="background-color: rgba(255, 255, 0, 1)"><span style="color: rgba(0, 0, 255, 1)">WHERE</span> <span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">b</span><span style="color: rgba(255, 0, 0, 1)">]</span>.<span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">author</span><span style="color: rgba(255, 0, 0, 1)">]</span> <span style="color: rgba(128, 128, 128, 1)">=</span> N<span style="color: rgba(255, 0, 0, 1)">'</span><span style="color: rgba(255, 0, 0, 1)">老周</span><span style="color: rgba(255, 0, 0, 1)">'</span></span></strong></pre>
</div>
<p> </p>
<p>下面来看看另一种应用情形——映射表值函数。</p>
<p>先在 SQL Server 中创建一个内联表值函数,名为 get_all_books,返回表中所有行。</p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 255, 1)">CREATE</span> <span style="color: rgba(0, 0, 255, 1)">FUNCTION</span> <span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">dbo</span><span style="color: rgba(255, 0, 0, 1)">]</span>.<span style="color: rgba(255, 0, 0, 1)">[</span><span style="color: rgba(255, 0, 0, 1)">get_all_books</span><span style="color: rgba(255, 0, 0, 1)">]</span><span style="color: rgba(0, 0, 0, 1)">()
</span><span style="color: rgba(0, 0, 255, 1)">RETURNS</span> <span style="color: rgba(0, 0, 255, 1)">TABLE</span>
<span style="color: rgba(0, 0, 255, 1)">RETURN</span> <span style="color: rgba(0, 0, 255, 1)">select</span> <span style="color: rgba(128, 128, 128, 1)">*</span> <span style="color: rgba(0, 0, 255, 1)">from</span> dbo.books;</pre>
</div>
<p>回到 .NET 项目,咱们要映射一下函数。</p>
<p>A、先在 DbContext 的派生类中定义一个方法,用于映射到函数,不需要实现方法体,直接抛异常就行。</p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 255, 1)">internal</span> IQueryable<Book><span style="color: rgba(0, 0, 0, 1)"><strong><span style="background-color: rgba(255, 255, 0, 1)"> GetAllBooksMap</span></strong>()
{
</span><span style="color: rgba(0, 0, 255, 1)">throw</span> <span style="color: rgba(0, 0, 255, 1)">new</span><span style="color: rgba(0, 0, 0, 1)"> NotSupportedException();
}</span></pre>
</div>
<p>实际上,EF Core 并不会真正调用方法,只是通过生成表达式树 + 反射出方法名,然后再找到与方法名对应的数据库中的函数罢了。所以,方法不需要实现代码。</p>
<p>B、OnModelCreating 方法要改一下,映射列名的 HasColumnName 方法不能在 ToTable 方法中配置,否则表值函数返回的实体不能正确映射。</p>
<div class="cnblogs_code">
<pre>modelBuilder.Entity<Book>(t =><span style="color: rgba(0, 0, 0, 1)">
{
t.ToTable(</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">books</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(0, 0, 0, 1)">);
t.HasKey(t </span>=><span style="color: rgba(0, 0, 0, 1)"> t.BookId);
<span style="background-color: rgba(255, 255, 0, 1)">t.Property(x </span></span><span style="background-color: rgba(255, 255, 0, 1)">=> x.BookId).HasColumnName(<span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">book_id</span><span style="color: rgba(128, 0, 0, 1)">"</span></span><span style="color: rgba(0, 0, 0, 1)"><span style="background-color: rgba(255, 255, 0, 1)">)</span>;
<span style="background-color: rgba(255, 255, 0, 1)">t.Property(z </span></span><span style="background-color: rgba(255, 255, 0, 1)">=> z.Title).HasColumnName(<span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">title</span><span style="color: rgba(128, 0, 0, 1)">"</span></span><span style="color: rgba(0, 0, 0, 1)"><span style="background-color: rgba(255, 255, 0, 1)">)</span>;
<span style="background-color: rgba(255, 255, 0, 1)">t.Property(k </span></span><span style="background-color: rgba(255, 255, 0, 1)">=> k.Description).HasColumnName(<span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">desc</span><span style="color: rgba(128, 0, 0, 1)">"</span></span><span style="color: rgba(0, 0, 0, 1)"><span style="background-color: rgba(255, 255, 0, 1)">)</span>;
<span style="background-color: rgba(255, 255, 0, 1)">t.Property(f </span></span><span style="background-color: rgba(255, 255, 0, 1)">=> f.Author).HasColumnName(<span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">author</span><span style="color: rgba(128, 0, 0, 1)">"</span></span><span style="color: rgba(0, 0, 0, 1)"><span style="background-color: rgba(255, 255, 0, 1)">)</span>;
});</span></pre>
</div>
<p>也就是列名映射要在 Property 上配置,不能在 TableBuilder 上配置。</p>
<p>C、HasDbFunction 映射函数。</p>
<div class="cnblogs_code">
<pre> <span style="color: rgba(0, 128, 0, 1)">//</span><span style="color: rgba(0, 128, 0, 1)"> 注意数据库中的函数名与类方法不同</span>
modelBuilder.HasDbFunction(GetType().GetMethod(nameof(<strong><span style="background-color: rgba(255, 255, 0, 1)">GetAllBooksMap</span></strong>), BindingFlags.NonPublic)!).HasName(<span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">get_all_books</span><span style="color: rgba(128, 0, 0, 1)">"</span>); </pre>
</div>
<p> </p>
<p>这里有个误区:很多大伙伴以为这样就完事了,然后就开始调用代码了。</p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 255, 1)">using</span> <span style="color: rgba(0, 0, 255, 1)">var</span> context = <span style="color: rgba(0, 0, 255, 1)">new</span><span style="color: rgba(0, 0, 0, 1)"> MyDbContext();
</span><span style="color: rgba(0, 0, 255, 1)">foreach</span>(Book bk <span style="color: rgba(0, 0, 255, 1)">in</span><span style="color: rgba(0, 0, 0, 1)"><span style="background-color: rgba(255, 255, 0, 1)"> context.GetAllBooksMap()</span>)
{
Console.WriteLine($</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">{bk.Author,-10}{bk.Title}</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(0, 0, 0, 1)">);
}</span></pre>
</div>
<p>你以为这样是对的,但运行后就是错的。上面不是说了吗?<em><span style="color: rgba(0, 0, 128, 1)">GetAllBooksMap 方法是没有实现的,你不能直接调用它!不能调用,不能调用,不能调用</span></em>!!</p>
<p>我们还需要再给 DbContext 的派生类再定义一个方法,使用 FromExpression 方法让 GetAllBooksMap 转为表达式树。</p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 255, 1)">public</span> IQueryable<Book><span style="color: rgba(0, 0, 0, 1)"> GetAllBooks()
{
</span><span style="color: rgba(0, 0, 255, 1)">return</span><span style="color: rgba(0, 0, 255, 1)">this</span>.FromExpression(() =><span style="color: rgba(0, 0, 0, 1)"><strong><span style="background-color: rgba(255, 255, 0, 1)"> GetAllBooksMap()</span></strong>);
}</span></pre>
</div>
<p>这么一来,GetAllBooksMap() 就成了表达式树,EF 不会真的调用它,只是获取相关信息,再翻译成 SQL。</p>
<p>然后这样用:</p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 255, 1)">using</span> <span style="color: rgba(0, 0, 255, 1)">var</span> context = <span style="color: rgba(0, 0, 255, 1)">new</span><span style="color: rgba(0, 0, 0, 1)"> MyDbContext();
</span><span style="color: rgba(0, 0, 255, 1)">foreach</span>(Book bk <span style="color: rgba(0, 0, 255, 1)">in</span><span style="color: rgba(0, 0, 0, 1)"> context.<strong><span style="background-color: rgba(255, 255, 0, 1)">GetAllBooks()</span></strong>)
{
Console.WriteLine($</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">{bk.Author,-10}{bk.Title}</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(0, 0, 0, 1)">);
}</span></pre>
</div>
<p>看,四条记录就读出来了。</p>
<p><img src="https://img2024.cnblogs.com/blog/367389/202510/367389-20251014222731451-737886574.png" alt="image" width="232" height="63" loading="lazy"></p>
<p>可是,你也发现了,这TM太麻烦了,为了表值函数映射,我要封装两个方法成员。其实,这里可以把两个方法合成一个:</p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 255, 1)">public</span> IQueryable<Book><span style="color: rgba(0, 0, 0, 1)"> GetAllBooks()
{
</span><span style="color: rgba(0, 0, 255, 1)">return</span><span style="color: rgba(0, 0, 255, 1)">this</span>.FromExpression(() =><span style="color: rgba(0, 0, 0, 1)"><strong><span style="background-color: rgba(255, 255, 0, 1)"> GetAllBooks</span></strong>());
}</span></pre>
</div>
<p>由于是公共方法,OnModelCreating 中的 HasDbFunction 代码也可以精简一下。</p>
<div class="cnblogs_code">
<pre>modelBuilder.HasDbFunction(GetType().GetMethod(nameof(GetAllBooks))!).HasName(<span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">get_all_books</span><span style="color: rgba(128, 0, 0, 1)">"</span>); </pre>
</div>
<p>这时候你又搞不懂了,What?GetAllBooks 方法怎么自己调用了自己?不不不,没有的事,你又忘了,FromExpression 只是转换为表达式树,并不会真的调用它。所以,这样合并后,其实代码是这样走的:</p>
<p>1、访问 context.GetAllBooks() ,这时候,GetAllBooks 方法确实被调用了,是你的代码调用的,不是EF调用;</p>
<p>2、GetAllBooks 方法被你调用后,FromExpression 方法被调用;</p>
<p>3、FromExpression 方法参数中,lambda 表达式虽然又引用了一次 GetAllBooks 方法,但这一次它不会被调用,EF Core 只是用来获取方法名。</p>
<p>现在明白了吗?</p>
<p>对,微软官方文档中的示例用的就是这种合并的方法,表面上看好像自己调用了自己,实则不会。</p>
<p>好了,今天就水到这里。</p>
<p> </p><br><br>
来源:https://www.cnblogs.com/tcjiaan/p/19136951
頁:
[1]