9.9、存储过程、视图、函数操作

存储过程、视图、函数操作是关系型数据库处理复杂逻辑的强大辅助。

存储过程

在Hoa Framework v1.4.0 版本新增了 切面上下文功能,也就是下面的代码不再推荐使用,见 9.12、切面上下文(TangentDbContext)章节。

无需配置 DbSet 方式调用存储过程(推荐)

正常情况下,存储过程也需要和实体一样,需要配置 DbSet无键实体类型,而且只能返回指定存储过程指定的返回实体模型,这样比较繁琐。(见 DbSet 方式

所以,Hoa Framework 提供了更加简便的方式。支持返回任意数据类型。代码如下:

// 返回存储过程实体模型
_testRepository.SqlProcedureQuery<PROC_Entity>(DbConsts.Procedure.PROC_NAME_KEY, new (){});

// 返回任意类型
_testRepository.SqlProcedureQuery<int>(DbConsts.Procedure.PROC_NAME_KEY, new (){})

这种方式最大的好处就是可使用任何仓储对象进行执行存储过程,是极速开发推荐方式

配置 DbSet 方式调用存储过程

在Hoa Framework v1.4.0 版本新增了 切面上下文功能,也就是下面的代码不再推荐使用,见 9.12、切面上下文(TangentDbContext)章节。

由于存储过程是无键实体类型,所以,需要手动配置 DbSet 和 创建存储过程返回值 Entity 模型。

在 Hoa Framework 框架中,需要手动配置在 Hoa.EntityFramework.Core.ManualHoaDbContext.cs 文件中,避免被 Database First 生成器每次生成后覆盖掉。代码如下:

using Microsoft.EntityFrameworkCore;
using System;
using Hoa.Core;

namespace Hoa.EntityFrameworkCore
{
    public partial class HoaDbContext
    {
        // PROC_Entity 是存储过程返回值对应的实体模型
        public virtual DbSet<PROC_Entity> PROC_Entities { get; set; }

        partial void OnModelCreatingPartial(ModelBuilder modelBuilder)
        {
            // 配置存储过程无键实体模型
            modelBuilder.Entity<PROC_Entity>(entity =>
            {
                entity.HasNoKey();
                // 通常我们需要将存储过程名称作为常量定义起来
                // 这里的存储过程名称配置在 Hoa.Core.DbConsts.cs 文件中
                entity.ToView(DbConsts.Procedure.PROC_NAME_KEY);
            });
        }
    }
}

通过上述配置后,即可通过 IRepository<PROC_Entity> 获得存储过程操作仓储对象,然后调用 FromSql 方式调用存储过程。如:

这种方式只能返回指定存储过程指定的返回实体模型。

存储过程参数说明

由于存储过程存在参数且参数名可能和参数模型不一致,所以 Hoa Framework 内置了强大存储过程参数配置特性。

默认情况下,我们可以通过 匿名对象 作为存储过程的参数,这种做法的好处就是无需创建一个 存储过程参数模型,但是缺点也显而易见,就是必须严格按照存储过程定义的参数排序进行传入。而且匿名对象属性名也必须和存储过程参数名一一对应。

所以,Hoa Framework 建议开发者使用强类型模型作为存储过程参数模型,这样无需考虑排序,又能指定存储过程参数名称。

🏴 示例一:参数模型和存储过程名字一致

🏴 示例二:指定存储过程的实际参数,在存储过程参数模型中设置

这种方式可维护性强,也利于拓展。

视图

视图是一张虚拟表,也是无键实体类型,所以,需要手动配置 DbSet 和 创建视图返回值 Entity 模型,这一点和存储过程配置无异。

配置视图DbSet

在 Hoa Framework 框架中,需要手动配置在 Hoa.EntityFramework.Core.ManualHoaDbContext.cs 文件中,避免被 Database First 生成器每次生成后覆盖掉。代码如下:

通过上述配置后,即可通过 IRepository<VIEW_Entity> 获得视图仓储对象,之后就可以使用仓储中除了 增删改 以外的操作了通常视图只用作查询和权限控制。如:

视图使用

更多视图仓储的操作可查看 9.5、查询操作 文档。

函数

在Hoa Framework v1.4.0 版本新增了 切面上下文功能,也就是下面的代码不再推荐使用,见 9.12、切面上下文(TangentDbContext)章节。

数据库中函数的调用非常简单,无需配置 DbSet 模型,也无需创建特定的 IRepository<> 仓储对象。

函数直接调用

注意:这种方式只能单纯的输出函数的返回值,并不能联合 sql 进行查询。

数据库内置函数使用

EF Core 为我们提供了很多常用的内置函数,可以在 Lambda 条件中使用,主要是通过 EF.Functions 调用,如:

这个语句使用了 EF.Functions.DateDiffHour 最终生成的 Sql 如下:

也就是生成了对应 DATEDIFF 函数。

更多内置的EF函数可查看官方文档

自定义标量函数

这个有区别与上面的 函数调用,上面的 函数调用 只能用于单纯的调用函数,并不能在 Lambda 语句中使用。

而通过自定义标量函数的方式,可以实现在 LambdaLinq 中调用。

假设我们数据库有一个自定义标量函数:

接下来

第一步:我们需要在 Hoa.Core.DbScalarFunctions.cs 中申明:

第二步:需要在 Hoa.EntityFramework.Core.ManualHoaDbContext.cs 中配置中初始化:

配置好上面两步后,就可以使用该函数了:

最终生成的 Sql 如下:

这种自定义标量函数除了可以在 Where 条件中调用,还能在 Select 结果集中使用。如:

最后更新于

这有帮助吗?