• 浅入 ABP 系列(6):数据库配置


    浅入 ABP 系列(6):数据库配置

    版权护体©作者:痴者工良,微信公众号转载文章需要 《NCC开源社区》同意。

    本系列的第五篇:https://www.cnblogs.com/whuanle/p/13061059.html

    因为这一部分属于 ASP.NET Core 的基础部分,ABP 没有封装,因此没啥要说的。

    这一篇我们将来学习如何在 ABP 中添加数据库配置以及划分一个简单数据库模块的结构,我们将使用 EFCore + Freesql 来搭建数据库模块。

    强烈推荐 Freesql!Freesql 是叶老师出品的 ORM 框架,现在属于 NCC 成员项目,Freesql 解决了我很多在日常开发中的痛点,并且其对业务开发的考虑和众多有些的拓展功能,实在令我爱不释手!

    AbpBase.Database 中,通过 Nuget 添加以下几个库:

    版本都是 1.9.0-preview0917,你可以使用最新版本的。

    Freesql									
    FreeSql.Provider.Sqlite
    FreeSql.Provider.SqlServer
    FreeSql.Provider.MySql
    

    创建标准的 EFCore 数据库上下文

    在 ABP 中,EFCore 上下文类需要继承 AbpDbContext<T>,整体编写方法跟继承 DbContext<T> 一致 ,接下来我们将一步步来讲解在 AbpBase 中如何添加 EFCore 功能。

    连接字符串

    ABP 中,可以在上下文类加上一个 ConnectionStringName 特性,然后在配置服务时,ABP 会自动为其配置连接字符串。

        [ConnectionStringName("Default")]
        public partial class DatabaseContext : AbpDbContext<DatabaseContext>
    

    Default 是一个标识,你也可以填写其他字符串标识。

    定义隔离的上下文

    首先,我们在 AbpBase.Database 模块中,创建两个文件夹:

    BaseData
    ExtensionData
    

    BaseData 目录用来存放基础表结构的上下文,ExtensionData 用来存放可能会拓展或者经常变动的表结构。

    在 BaseData 中创建一个 AbpBaseDataContext 类,其内容如下:

    using Microsoft.EntityFrameworkCore;
    using Volo.Abp.Data;
    using Volo.Abp.EntityFrameworkCore;
    
    namespace AbpBase.Database
    {
        /// <summary>
        /// 上下文
        /// <para>这部分用于定义和配置基础表的映射</para>
        /// </summary>
        [ConnectionStringName("Default")]
        public partial class AbpBaseDataContext : AbpDbContext<AbpBaseDataContext>
        {
    
            #region 定义 DbSet<T>
    
            #endregion
    
    
            public AbpBaseDataContext(DbContextOptions<AbpBaseDataContext> options)
        : base(options)
            {
            }
    
            /// <summary>
            /// 定义映射
            /// </summary>
            /// <param name="modelBuilder"></param>
            protected override void OnModelCreating(ModelBuilder modelBuilder)
            {
                #region 定义 映射
    
                #endregion
    
                OnModelCreatingPartial(modelBuilder);
            }
    
            partial void OnModelCreatingPartial(ModelBuilder modelBuilder);
        }
    }
    

    在 ExtensionData 中也创建一个相同的 AbpBaseDataContext 类,其内容如下:

    using Microsoft.EntityFrameworkCore;
    
    namespace AbpBase.Database
    {
    
        public partial class AbpBaseDataContext
        {
            #region 定义 DbSet<T>
    
            #endregion
    
            /// <summary>
            /// 定义映射
            /// </summary>
            /// <param name="modelBuilder"></param>
            partial void OnModelCreatingPartial(ModelBuilder modelBuilder)
            {
    
            }
        }
    }
    

    分部类,前者用于定义那些非常基础的,程序核心的实体(表)以及映射。而后者定义后续可能多次修改的,设计时感觉有设计余地的。

    多数据库支持和配置

    这里我们将对上下文进行配置和注入,使得程序能够支持多数据库。

    AbpBase.Domain.Shared 项目中,创建一个枚举,其内容如下:

    namespace AbpBase.Domain.Shared
    {
        public enum AbpBaseDataType
        {
            Sqlite = 0,
            Mysql = 1,
            Sqlserver = 2
    
            // 其他数据库
        }
    }
    

    再创建一个 WholeShared 类,其内容如下:

    namespace AbpBase.Domain.Shared
    {
        /// <summary>
        /// 全局共享内容
        /// </summary>
        public static class WholeShared
        {
            // 数据库连接属性可以自行在配置文件中定义,这里写固定的,只是为了演示
    
            /// <summary>
            /// 数据库连接字符串
            /// </summary>
            public static readonly string SqlConnectString = "";
    
            /// <summary>
            /// 要使用的数据库类型
            /// </summary>
            public static readonly AbpBaseDataType DataType = AbpBaseDataType.Sqlite;
        }
    }
    

    然后我们在 AbpBaseDatabaseModule 模块中的 ConfigureServices 函数里面添加依赖注入:

    context.Services.AddAbpDbContext<AbpBaseDataContext>();
    

    这里不需要配置数据库连接字符串,后面可以通过 ABP 的一些方法来配置。

    配置上下文连接字符串

                string connectString = default;
                Configure<AbpDbConnectionOptions>(options =>
                {
                    connectString = WholeShared.SqlConnectString;
                    options.ConnectionStrings.Default = connectString;
                });
    

    配置多数据库支持:

                FreeSql.DataType dataType = default;
    
                Configure<AbpDbContextOptions>(options =>
                {
                    switch (WholeShared.DataType)
                    {
                        case AbpBaseDataType.Sqlite:
                            options.UseSqlite<AbpBaseDataContext>(); dataType = FreeSql.DataType.Sqlite; break;
                        case AbpBaseDataType.Mysql:
                            options.UseMySQL<AbpBaseDataContext>(); dataType = FreeSql.DataType.MySql; break;
                        case AbpBaseDataType.Sqlserver:
                            options.UseSqlServer<AbpBaseDataContext>(); dataType = FreeSql.DataType.SqlServer; break;
                    }
                });
    

    这样就完成了对 EFCore 的多数据库配置了。

    下面我们来使用类似的方法配置 Freesql。

    Freesql 配置服务

    首先,Freesql 里面有多种配置方式,例如 DbContext,读者可以到 Wiki 去学习 Freesqlhttps://github.com/dotnetcore/FreeSql/wiki/%E5%85%A5%E9%97%A8

    笔者这里使用的是 “非正规” 的设计方式,哈哈哈哈。

    BaseData 目录中,创建一个 FreesqlContext 类,其内容如下:

    using FreeSql.Internal;
    using System;
    using System.Collections.Generic;
    using System.Text;
    
    namespace AbpBase.Database
    {
        /// <summary>
        /// Freesql 上下文
        /// </summary>
        public partial class FreesqlContext
        {
            public static IFreeSql FreeselInstance => Freesql_Instance;
            private static IFreeSql Freesql_Instance;
    
            public static void Init(string connectStr, FreeSql.DataType dataType = FreeSql.DataType.Sqlite)
            {
                Freesql_Instance = new FreeSql.FreeSqlBuilder()
                    .UseNameConvert(NameConvertType.PascalCaseToUnderscore)
                    .UseConnectionString(dataType, connectStr)
    
                    //.UseAutoSyncStructure(true) // 自动同步实体结构到数据库,生产环境禁止使用!
    
                    .Build();
                OnModelCreating(Freesql_Instance);
            }
    
            private static void OnModelCreating(IFreeSql freeSql)
            {
    
    
                OnModelCreatingPartial(freeSql);
            }
        }
    }
    
    

    ExtensionData 目录中,创建 FreesqlContext 类 如下:

    using FreeSql;
    using System;
    using System.Collections.Generic;
    using System.Text;
    
    namespace AbpBase.Database
    {
        public partial class FreesqlContext
        {
            private static void OnModelCreatingPartial(IFreeSql freeSql)
            {
                var modelBuilder = freeSql.CodeFirst;
    
                SyncStruct(modelBuilder);
    
            }
    
            /// <summary>
            /// 同步结构到数据中
            /// </summary>
            /// <param name="codeFirst"></param>
            private static void SyncStruct(ICodeFirst codeFirst)
            {
                //  codeFirst.SyncStructure(typeof(user));
            }
        }
    }
    

    然后在 AbpBaseDatabaseModuleConfigureServices 函数中添加注入服务:

                FreesqlContext.Init(connectString, dataType);
                context.Services.AddSingleton(typeof(IFreeSql), FreesqlContext.FreeselInstance);
                context.Services.AddTransient(typeof(FreesqlContext), typeof(FreesqlContext));
    

    通过以上步骤,我们的 ABP 就可以支持多数据库了,EFCore + Freesql,并且将将表分级隔离维护。

  • 相关阅读:
    Kafka与RabbitMQ区别
    Illegal instruction 问题的解决方法
    Debian 6 , 十个串口为什么只识别到了 6个 剩下4 个被禁止了
    微信二次认证 C#
    修改XtraMessageBox的内容字体大小
    svn: E155017: Checksum mismatch while updating 校验错误的解决方法
    再生龙恢复分区后修复引导或debian linux修复引导 三部曲
    Clonezilla制作镜像时报错: errextfsclone.c:bitmap free count err
    我用windows live Writer 写个日志试试看
    Debian下签名无法验证
  • 原文地址:https://www.cnblogs.com/whuanle/p/13680005.html
Copyright © 2020-2023  润新知