搜索

查看: 3100|回复: 11

[ASP.NET] .net如何优雅的使用EFCore实例详解

[复制链接]
发表于 2023-5-4 11:26:04 | 显示全部楼层 |阅读模式
Editor 2023-5-4 11:26:04 3100 11 看全部
目录
  • 正文
  • DBSet清除计划
  • IEntityTypeConfiguration(表配置)
  • Repository(仓储)
  • Autofac
  • 数据库配置
  • 项目架构和源码
    正文
    EFCore是微软官方的一款ORM框架,主要是用于实体和数据库对象之间的操作。功能非常强大,在老版本的时候叫做EF,后来.net core问世,EFCore也随之问世。
    本文我们将用一个控制台项目Host一个web服务,并且使用本地Mysql作为数据库,使用EFCore的Code First模式进行数据操作。

    DBSet清除计划
    以前使用EF/EFCore的开发者应该都记得,需要在DBContext里写好多DBSet,一个表对应一个DBSet,然后在其他地方操作这些DBSet对相关的表进行增删改查。作为一个开发,这些重复操作都是我们希望避免的,我们可以利用反射机制将这些类型通过框架自带的方法循环注册进去。
    1.EF实体继承统一的接口,方便我们反射获取所有EF实体,接口可以设置一个泛型,来泛化我们的主键类型,因为可能存在不同的表的主键类型也不一样。
    统一的EF实体接口
    public interface IEFEntity
    {
        public TKey Id { get; set; }
    }
    统一的接口实现类
    public abstract class AggregateRoot : IEFEntity
    {
        public TKey Id { get; set; }
    }
    用户实体类
    public class User : AggregateRoot
    {
        public string UserName { get; set; }
        public DateTime Birthday { get; set; }
        public virtual ICollection[B] Books { get; set; }
    }
    2.利用反射获取某个程序集下所有的实体类
    public class EFEntityInfo
    {
        public (Assembly Assembly, IEnumerable Types) EFEntitiesInfo => (GetType().Assembly, GetEntityTypes(GetType().Assembly));
        private IEnumerable GetEntityTypes(Assembly assembly)
        {
            //获取当前程序集下所有的实现了IEFEntity的实体类
            var efEntities = assembly.GetTypes().Where(m => m.FullName != null
                                                            && Array.Exists(m.GetInterfaces(), t => t.IsGenericType && t.GetGenericTypeDefinition() == typeof(IEFEntity))
                                                            && !m.IsAbstract && !m.IsInterface).ToArray();
            return efEntities;
        }
    }
    3.DBContext实现类中OnModelCreating方法中注册这些类型
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        //循环实体类型,并且通过Entity方法注册类型
        foreach (var entityType in Types)
        {
            modelBuilder.Entity(entityType);
        }
        base.OnModelCreating(modelBuilder);
    }
    至此为止所有的实体类都被注册到DBContext中作为DBSets,再也不需要一个个写DBSet了,可以用过DbContext.Set[U]()获取用户的DBSet。

    IEntityTypeConfiguration(表配置)
    用数据库创建过表的同学都知道,在设计表的时候,可以给表添加很多配置和约束,在Code First模式中,很多同学都是在对象中通过注解的方式配置字段。如下就配置了用户名是不能为NULL和最大长度为500
    [Required]
    [MaxLength(500)]
    public string UserName { get; set; }
    也有的同学在DbContext中的OnModelCreating方法配置
    modelBuilder.Entity[U]().Property(x => x.UserName).IsRequired();
    这两种方法,前者入侵行太强,直接代码耦合到实体类中了,后者不够清楚,把一大堆表的配置写在一个方法里,当然了很多人说可以拆分不同的方法或者使用注释分开。但是!不够优雅!
    我们可以使用IEntityTypeConfiguration接口实现我们所想的优雅的表配置。
    1.创建一个配置基类,继承自IEntityTypeConfiguration,做一些通用的配置,比如设置主键,一般都是id啦,还有软删除等。
    public abstract class EntityTypeConfiguration : IEntityTypeConfiguration
           where TEntity : AggregateRoot
    {
        public virtual void Configure(EntityTypeBuilder builder)
        {
            var entityType = typeof(TEntity);
            builder.HasKey(x => x.Id);
            if (typeof(ISoftDelete).IsAssignableFrom(entityType))
            {
                builder.HasQueryFilter(d => EF.Property(d, "IsDeleted") == false);
            }
        }
    }
    2.创建用户实体/表独有的配置,比如设置用户名的最大长度,以及seed一些数据
    public class UserConfig : EntityTypeConfiguration[U]
    {
        public override void Configure(EntityTypeBuilder[U] builder)
        {
            base.Configure(builder);
            builder.Property(x => x.UserName).HasMaxLength(50);
            //mock一条数据
            builder.HasData(new User()
            {
                Id = "090213204",
                UserName = "Bruce",
                Birthday = DateTime.Parse("1996-08-24")
            });
        }
    }
    当然还有很多配置可以设置,比如索引,导航属性,唯一键等。如下图书实体
    public class BookConfig : EntityTypeConfiguration[B]
    {
        public override void Configure(EntityTypeBuilder[B] builder)
        {
            base.Configure(builder);
            builder.Property(x => x.Id).ValueGeneratedOnAdd(); //设置book的id自增
            builder.Property(x => x.BookName).HasMaxLength(500).IsRequired();
            builder.HasIndex(x => x.Author);//作者添加索引
            builder.HasIndex(x => x.SN).IsUnique();//序列号添加唯一索引
            builder.HasOne(r => r.User).WithMany(x=>x.Books)
                .HasForeignKey(r => r.UserId).IsRequired();//导航属性,本质就是创建外键,虽然查询很方便,生产中不建议使用!!!
        }
    }
    3.DBContext中应用配置
    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.HasCharSet("utf8mb4 ");
        var (Assembly, Types) = _efEntitysInfo.EFEntitiesInfo;
        foreach (var entityType in Types)
        {
            modelBuilder.Entity(entityType);
        }
        //只需要将配置类所在的程序集给到,它会自动加载
        modelBuilder.ApplyConfigurationsFromAssembly(Assembly);
        base.OnModelCreating(modelBuilder);
    }

    Repository(仓储)
    这个不过分介绍,特别是基于http的微服务中基本都有这个。
    1.创建一个仓储基类,对于不同的实体,创建一样的增删改查方法。
    简单写几个查询的方法定义。
    public interface IAsyncRepository where TEntity : class
    {
        IQueryable All();
        IQueryable All(string[] propertiesToInclude);
        IQueryable Where(Expression> filter);
        IQueryable Where(Expression> filter, string[] propertiesToInclude);
    }
    2.创建仓储实现类,将DBContext注入到构造中
    public class GenericRepository : IAsyncRepository where TEntity : class
    {
        protected readonly LibraryDbContext _dbContext;
        public GenericRepository(LibraryDbContext dbContext)
        {
            _dbContext = dbContext;
        }
        ~GenericRepository()
        {
            _dbContext?.Dispose();
        }
        public virtual IQueryable All()
        {
            return All(null);
        }
        public virtual IQueryable All(string[] propertiesToInclude)
        {
            var query = _dbContext.Set().AsNoTracking();
            if (propertiesToInclude != null)
            {
                foreach (var property in propertiesToInclude.Where(p => !string.IsNullOrWhiteSpace(p)))
                {
                    query = query.Include(property);
                }
            }
            return query;
        }
    }

    Autofac
    1.注入DBContext到Repository的构造方法中,并且注入Repository
    public class EFCoreEleganceUseEFCoreModule : Module
    {
        protected override void Load(ContainerBuilder builder)
        {
            base.Load(builder);
            builder.RegisterModule(); //注入domain模块
            builder.RegisterGeneric(typeof(GenericRepository))//将dbcontext注入到仓储的构造中
                    .UsingConstructor(typeof(LibraryDbContext))
                    .AsImplementedInterfaces()
                    .InstancePerDependency();
            builder.RegisterType().As[I]().InstancePerDependency();
        }
    }
    2.Domain注入EFEntityInfo
    public class EFCoreEleganceUseDomainModule : Module
    {
        protected override void Load(ContainerBuilder builder)
        {
            builder.RegisterType().SingleInstance();
        }
    }

    数据库配置
    1.注入DBContext,从配置文件读取数据库配置,然后根据开发/生产环境做一些特殊处理
    var mysqlConfig = hostContext.Configuration.GetSection("Mysql").Get();
    var serverVersion = new MariaDbServerVersion(new Version(mysqlConfig.Version));
    services.AddDbContextFactory(options =>
    {
        options.UseMySql(mysqlConfig.ConnectionString, serverVersion, optionsBuilder =>
        {
            optionsBuilder.MinBatchSize(4);
            optionsBuilder.CommandTimeout(10);
            optionsBuilder.MigrationsAssembly(mysqlConfig.MigrationAssembly);//迁移文件所在的程序集
            optionsBuilder.UseQuerySplittingBehavior(QuerySplittingBehavior.SplitQuery);
        }).UseQueryTrackingBehavior(QueryTrackingBehavior.NoTracking);
        //开发环境可以打开日志记录和显示详细的错误
        if (hostContext.HostingEnvironment.IsDevelopment())
        {
            options.EnableSensitiveDataLogging();
            options.EnableDetailedErrors();
        }
    });

    项目架构和源码

    20221128084747033.png

    20221128084747033.png


    项目只是一个demo架构,并不适用于生产,主程序是一个控制台项目,只需要引用相关的包和模块,就可以启动一个web host.
    全部代码已经全部上传到github:https://github.com/BruceQiu1996/EFCoreDemo该项目是一个可以启动运行的基于.net6的控制台项目,启动后会启动一个web host和一个swagger页面。
    以上就是.net如何优雅的使用EFCore实例详解的详细内容,更多关于.net使用EFCore的资料请关注知鸟论坛其它相关文章!
  • 回复

    使用道具 举报

    发表于 2023-6-28 22:47:22 | 显示全部楼层
    无人岛屿颈 2023-6-28 22:47:22 看全部
    既然你诚信诚意的推荐了,那我就勉为其难的看看吧!知鸟论坛不走平凡路。
    回复

    使用道具 举报

    发表于 2023-6-29 18:58:20 | 显示全部楼层
    123456865 2023-6-29 18:58:20 看全部
    楼主太厉害了!楼主,I*老*虎*U!我觉得知鸟论坛真是个好地方!
    回复

    使用道具 举报

    发表于 2023-6-29 21:43:03 | 显示全部楼层
    永远就三年疗 2023-6-29 21:43:03 看全部
    我看不错噢 谢谢楼主!知鸟论坛越来越好!
    回复

    使用道具 举报

    发表于 2023-6-30 00:05:08 | 显示全部楼层
    永远爱你冰塘 2023-6-30 00:05:08 看全部
    楼主,大恩不言谢了!知鸟论坛是最棒的!
    回复

    使用道具 举报

    发表于 2023-6-30 10:06:04 | 显示全部楼层
    幸福341 2023-6-30 10:06:04 看全部
    这个帖子不回对不起自己!我想我是一天也不能离开知鸟论坛
    回复

    使用道具 举报

    发表于 2023-6-30 11:58:13 | 显示全部楼层
    墙和鸡蛋 2023-6-30 11:58:13 看全部
    其实我一直觉得楼主的品味不错!呵呵!知鸟论坛太棒了!
    回复

    使用道具 举报

    发表于 2023-6-30 23:59:22 | 显示全部楼层
    计划你大爷计j 2023-6-30 23:59:22 看全部
    楼主发贴辛苦了,谢谢楼主分享!我觉得知鸟论坛是注册对了!
    回复

    使用道具 举报

    发表于 2023-7-4 11:32:09 | 显示全部楼层
    素色流年783 2023-7-4 11:32:09 看全部
    楼主发贴辛苦了,谢谢楼主分享!我觉得知鸟论坛是注册对了!
    回复

    使用道具 举报

    发表于 2023-7-4 12:54:27 | 显示全部楼层
    462710480 2023-7-4 12:54:27 看全部
    楼主太厉害了!楼主,I*老*虎*U!我觉得知鸟论坛真是个好地方!
    回复

    使用道具 举报

    • 您可能感兴趣
    点击右侧快捷回复 【请勿灌水】
    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则 返回列表

    RSS订阅| SiteMap| 小黑屋| 知鸟论坛
    联系邮箱E-mail:zniao@foxmail.com
    快速回复 返回顶部 返回列表