日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

.net如何優雅的使用EFCore實例詳解

瀏覽:337日期:2022-06-09 09:34:40
目錄
  • 正文
    • 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<TKey>{    public TKey Id { get; set; }}

統一的接口實現類

public abstract class AggregateRoot<TKey> : IEFEntity<TKey>{    public TKey Id { get; set; }}

用戶實體類

public class User : AggregateRoot<string>{    public string UserName { get; set; }    public DateTime Birthday { get; set; }    public virtual ICollection<Book> Books { get; set; }}

2.利用反射獲取某個程序集下所有的實體類

public class EFEntityInfo{    public (Assembly Assembly, IEnumerable<Type> Types) EFEntitiesInfo => (GetType().Assembly, GetEntityTypes(GetType().Assembly));    private IEnumerable<Type> 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<User>()獲取用戶的DBSet。

IEntityTypeConfiguration(表配置)

用數據庫創建過表的同學都知道,在設計表的時候,可以給表添加很多配置和約束,在Code First模式中,很多同學都是在對象中通過注解的方式配置字段。如下就配置了用戶名是不能為NULL和最大長度為500

[Required][MaxLength(500)]public string UserName { get; set; }

也有的同學在DbContext中的OnModelCreating方法配置

modelBuilder.Entity<User>().Property(x => x.UserName).IsRequired();

這兩種方法,前者入侵行太強,直接代碼耦合到實體類中了,后者不夠清楚,把一大堆表的配置寫在一個方法里,當然了很多人說可以拆分不同的方法或者使用注釋分開。但是!不夠優雅!
我們可以使用IEntityTypeConfiguration接口實現我們所想的優雅的表配置。
1.創建一個配置基類,繼承自IEntityTypeConfiguration,做一些通用的配置,比如設置主鍵,一般都是id啦,還有軟刪除等。

public abstract class EntityTypeConfiguration<TEntity, TKey> : IEntityTypeConfiguration<TEntity>       where TEntity : AggregateRoot<TKey>{    public virtual void Configure(EntityTypeBuilder<TEntity> builder)    {var entityType = typeof(TEntity);builder.HasKey(x => x.Id);if (typeof(ISoftDelete).IsAssignableFrom(entityType)){    builder.HasQueryFilter(d => EF.Property<bool>(d, "IsDeleted") == false);}    }}

2.創建用戶實體/表獨有的配置,比如設置用戶名的最大長度,以及seed一些數據

public class UserConfig : EntityTypeConfiguration<User, string>{    public override void Configure(EntityTypeBuilder<User> 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<Book, long>{    public override void Configure(EntityTypeBuilder<Book> 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<TEntity, Tkey> where TEntity : class{    IQueryable<TEntity> All();    IQueryable<TEntity> All(string[] propertiesToInclude);    IQueryable<TEntity> Where(Expression<Func<TEntity, bool>> filter);    IQueryable<TEntity> Where(Expression<Func<TEntity, bool>> filter, string[] propertiesToInclude);}

2.創建倉儲實現類,將DBContext注入到構造中

public class GenericRepository<TEntity, Tkey> : IAsyncRepository<TEntity, Tkey> where TEntity : class{    protected readonly LibraryDbContext _dbContext;    public GenericRepository(LibraryDbContext dbContext)    {_dbContext = dbContext;    }    ~GenericRepository()    {_dbContext?.Dispose();    }    public virtual IQueryable<TEntity> All()    {return All(null);    }    public virtual IQueryable<TEntity> All(string[] propertiesToInclude)    {var query = _dbContext.Set<TEntity>().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<EFCoreEleganceUseDomainModule>(); //注入domain模塊builder.RegisterGeneric(typeof(GenericRepository<,>))//將dbcontext注入到倉儲的構造中.UsingConstructor(typeof(LibraryDbContext)).AsImplementedInterfaces().InstancePerDependency();builder.RegisterType<WorkUnit>().As<IWorkUnit>().InstancePerDependency();    }}

2.Domain注入EFEntityInfo

public class EFCoreEleganceUseDomainModule : Module{    protected override void Load(ContainerBuilder builder)    {builder.RegisterType<EFEntityInfo>().SingleInstance();    }}

數據庫配置

1.注入DBContext,從配置文件讀取數據庫配置,然后根據開發/生產環境做一些特殊處理

var mysqlConfig = hostContext.Configuration.GetSection("Mysql").Get<MysqlOptions>();var serverVersion = new MariaDbServerVersion(new Version(mysqlConfig.Version));services.AddDbContextFactory<LibraryDbContext>(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();    }});

項目架構和源碼

項目只是一個demo架構,并不適用于生產,主程序是一個控制臺項目,只需要引用相關的包和模塊,就可以啟動一個web host.

全部代碼已經全部上傳到github:https://github.com/BruceQiu1996/EFCoreDemo該項目是一個可以啟動運行的基于.net6的控制臺項目,啟動后會啟動一個web host和一個swagger頁面。

以上就是.net如何優雅的使用EFCore實例詳解的詳細內容,更多關于.net使用EFCore的資料請關注其它相關文章!

標簽: ASP.NET
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
久久成人av| 亚洲神马久久| 日韩不卡一区二区| 欧美国产极品| 香蕉成人久久| 精品日韩毛片| 国产精品亲子伦av一区二区三区| 精品一区二区三区的国产在线观看| 美女国产一区二区三区| 色综合视频一区二区三区日韩| 国产剧情一区二区在线观看| 亚洲精品a级片| 久久久久久久久久久9不雅视频| 亚洲精品国产偷自在线观看| 日韩午夜视频在线| 久久中文亚洲字幕| 日韩三级精品| 精品国产黄a∨片高清在线| 在线精品国产亚洲| 国产成人免费视频网站视频社区| 国产一级一区二区| 久久精品资源| 国产激情欧美| 日韩国产91| 午夜精品亚洲| 欧美日韩中文一区二区| 欧美激情日韩| 久久99精品久久久野外观看| 99xxxx成人网| 色狠狠一区二区三区| 91九色综合| 日本一区二区高清不卡| 日本亚洲最大的色成网站www| 久久精品青草| 蜜桃伊人久久| 国产字幕视频一区二区| 91麻豆国产自产在线观看亚洲| 久久不卡日韩美女| 亚洲成人va| 成人午夜精品| 亚洲五月综合| 亚洲69av| 麻豆精品视频在线观看免费| 精品一区不卡| 久久精品国产亚洲夜色av网站| 国产一卡不卡| 999精品在线| 在线观看免费一区二区| 视频一区欧美日韩| 欧美高清不卡| 色8久久久久| 婷婷成人在线| 日韩中文字幕高清在线观看| 视频一区中文字幕精品| 日韩精品永久网址| 国产精品成人3p一区二区三区| 国产精品毛片久久久| 国产精久久久| 视频一区视频二区在线观看| 日韩不卡一区二区三区| 影音先锋久久精品| 高清精品久久| 999国产精品视频| 国产精品密蕾丝视频下载| 日韩精品欧美精品| 中文字幕成人| 欧美日韩国产在线观看网站 | 欧美日韩 国产精品| 亚洲在线成人| 国产亚洲一区二区手机在线观看 | 精品伊人久久久| 天堂俺去俺来也www久久婷婷| 欧美日韩国产精品一区二区亚洲| 国产亚洲激情| 日韩午夜一区| 国产精品一区二区av交换| 精品国产成人| 免费日韩成人| 日韩一区二区三区四区五区| 日韩一区二区免费看| 精品国产中文字幕第一页| 综合视频一区| 国产区精品区| 性感美女一区二区在线观看| 97精品97| 99视频在线精品国自产拍免费观看| 亚洲天堂av资源在线观看| 久久亚洲欧美| 美女国产精品久久久| 在线一区av| 蜜臀精品久久久久久蜜臀| 亚洲二区视频| 亚洲精品黄色| 视频在线在亚洲| 视频一区二区三区入口| 影音先锋国产精品| 亚洲女同中文字幕| 久久99影视| 久久亚洲道色| 日韩欧美一区二区三区免费观看| 成人在线免费观看网站| 美腿丝袜亚洲一区| 精品美女久久| 国产精品精品| 日韩精品电影一区亚洲| 日本在线成人| 国产传媒在线| 国产日韩三级| 夜夜嗨一区二区| 黄色精品视频| 精品视频国产| 欧美国产不卡| 欧美aa一级| 欧美在线黄色| 图片区亚洲欧美小说区| 麻豆成人av在线| 国产一区日韩一区| 国产精品永久| 欧美不卡高清| 精品视频亚洲| 丝瓜av网站精品一区二区| 日韩精品免费一区二区夜夜嗨| 国产精品久久久久久妇女| 精品久久电影| 青青国产91久久久久久| av高清不卡| 精品国产欧美日韩一区二区三区| 欧美日韩1区| 六月天综合网| 日韩激情中文字幕| 老鸭窝毛片一区二区三区| 天堂成人国产精品一区| 久久国产生活片100| 五月天激情综合网| 成人精品亚洲| 欧美天堂视频| 蜜桃久久久久久| 精品国产欧美| 国产+成+人+亚洲欧洲在线| 91精品国产自产在线丝袜啪| 久久国产三级精品| 麻豆久久一区| 午夜国产精品视频| 国产伊人久久| 久久wwww| 亚洲精品三级| 另类中文字幕国产精品| 免费亚洲婷婷| 成人久久久久| 国产精品主播在线观看| 国产精品一站二站| 91国内精品| 日韩国产在线不卡视频| 日本aⅴ亚洲精品中文乱码| 亚洲一区不卡| 日韩中文字幕| 国产欧美另类| 视频一区视频二区中文字幕| 亚洲免费观看高清完整版在线观| 国产精品7m凸凹视频分类| 丁香婷婷久久| 久久国产福利| 在线看片日韩| 麻豆成人在线| 精品国产三区在线| 国产亚洲电影| 久久女人天堂| 青青草精品视频| 日韩美女一区二区三区在线观看| 亚洲欧美网站在线观看| 国内精品亚洲| 欧美日韩日本国产亚洲在线| 国产精品主播| 99视频精品全国免费| 精品一区二区三区四区五区| 欧美一区二区三区久久精品| 国产亚洲一区| 亚洲精品大片| 国产精品成人a在线观看| 欧洲av不卡| 青草国产精品| 色爱综合av| 国产激情久久| 国产在线视频欧美一区| av一区二区高清| 日韩中文在线播放| 日韩亚洲精品在线观看| 日韩中文字幕91| 日韩高清中文字幕一区二区| 麻豆精品视频在线| 亚洲一区欧美二区| 亚洲另类av| 视频一区二区国产| 在线综合亚洲| caoporn视频在线| 日本欧美韩国一区三区| 亚洲四虎影院| 国产精品色在线网站| 91精品综合| 午夜免费一区|