成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

.net如何優(yōu)雅的使用EFCore實(shí)例詳解

瀏覽:282日期:2022-06-09 09:34:40
目錄
  • 正文
    • DBSet清除計(jì)劃
    • IEntityTypeConfiguration(表配置)
    • Repository(倉(cāng)儲(chǔ))
    • Autofac
    • 數(shù)據(jù)庫(kù)配置
    • 項(xiàng)目架構(gòu)和源碼

正文

EFCore是微軟官方的一款ORM框架,主要是用于實(shí)體和數(shù)據(jù)庫(kù)對(duì)象之間的操作。功能非常強(qiáng)大,在老版本的時(shí)候叫做EF,后來(lái).net core問(wèn)世,EFCore也隨之問(wèn)世。

本文我們將用一個(gè)控制臺(tái)項(xiàng)目Host一個(gè)web服務(wù),并且使用本地Mysql作為數(shù)據(jù)庫(kù),使用EFCore的Code First模式進(jìn)行數(shù)據(jù)操作。

DBSet清除計(jì)劃

以前使用EF/EFCore的開(kāi)發(fā)者應(yīng)該都記得,需要在DBContext里寫好多DBSet,一個(gè)表對(duì)應(yīng)一個(gè)DBSet,然后在其他地方操作這些DBSet對(duì)相關(guān)的表進(jìn)行增刪改查。作為一個(gè)開(kāi)發(fā),這些重復(fù)操作都是我們希望避免的,我們可以利用反射機(jī)制將這些類型通過(guò)框架自帶的方法循環(huán)注冊(cè)進(jìn)去。

1.EF實(shí)體繼承統(tǒng)一的接口,方便我們反射獲取所有EF實(shí)體,接口可以設(shè)置一個(gè)泛型,來(lái)泛化我們的主鍵類型,因?yàn)榭赡艽嬖诓煌谋淼闹麈I類型也不一樣。

統(tǒng)一的EF實(shí)體接口

public interface IEFEntity<TKey>{    public TKey Id { get; set; }}

統(tǒng)一的接口實(shí)現(xiàn)類

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

用戶實(shí)體類

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

2.利用反射獲取某個(gè)程序集下所有的實(shí)體類

public class EFEntityInfo{    public (Assembly Assembly, IEnumerable<Type> Types) EFEntitiesInfo => (GetType().Assembly, GetEntityTypes(GetType().Assembly));    private IEnumerable<Type> GetEntityTypes(Assembly assembly)    {//獲取當(dāng)前程序集下所有的實(shí)現(xiàn)了IEFEntity的實(shí)體類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實(shí)現(xiàn)類中OnModelCreating方法中注冊(cè)這些類型

protected override void OnModelCreating(ModelBuilder modelBuilder){    //循環(huán)實(shí)體類型,并且通過(guò)Entity方法注冊(cè)類型    foreach (var entityType in Types)    {modelBuilder.Entity(entityType);    }    base.OnModelCreating(modelBuilder);}

至此為止所有的實(shí)體類都被注冊(cè)到DBContext中作為DBSets,再也不需要一個(gè)個(gè)寫DBSet了,可以用過(guò)DbContext.Set<User>()獲取用戶的DBSet。

IEntityTypeConfiguration(表配置)

用數(shù)據(jù)庫(kù)創(chuàng)建過(guò)表的同學(xué)都知道,在設(shè)計(jì)表的時(shí)候,可以給表添加很多配置和約束,在Code First模式中,很多同學(xué)都是在對(duì)象中通過(guò)注解的方式配置字段。如下就配置了用戶名是不能為NULL和最大長(zhǎng)度為500

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

也有的同學(xué)在DbContext中的OnModelCreating方法配置

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

這兩種方法,前者入侵行太強(qiáng),直接代碼耦合到實(shí)體類中了,后者不夠清楚,把一大堆表的配置寫在一個(gè)方法里,當(dāng)然了很多人說(shuō)可以拆分不同的方法或者使用注釋分開(kāi)。但是!不夠優(yōu)雅!
我們可以使用IEntityTypeConfiguration接口實(shí)現(xiàn)我們所想的優(yōu)雅的表配置。
1.創(chuàng)建一個(gè)配置基類,繼承自IEntityTypeConfiguration,做一些通用的配置,比如設(shè)置主鍵,一般都是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.創(chuàng)建用戶實(shí)體/表獨(dú)有的配置,比如設(shè)置用戶名的最大長(zhǎng)度,以及seed一些數(shù)據(jù)

public class UserConfig : EntityTypeConfiguration<User, string>{    public override void Configure(EntityTypeBuilder<User> builder)    {base.Configure(builder);builder.Property(x => x.UserName).HasMaxLength(50);//mock一條數(shù)據(jù)builder.HasData(new User(){    Id = "090213204",    UserName = "Bruce",    Birthday = DateTime.Parse("1996-08-24")});    }}

當(dāng)然還有很多配置可以設(shè)置,比如索引,導(dǎo)航屬性,唯一鍵等。如下圖書(shū)實(shí)體

public class BookConfig : EntityTypeConfiguration<Book, long>{    public override void Configure(EntityTypeBuilder<Book> builder)    {base.Configure(builder);builder.Property(x => x.Id).ValueGeneratedOnAdd(); //設(shè)置book的id自增builder.Property(x => x.BookName).HasMaxLength(500).IsRequired();builder.HasIndex(x => x.Author);//作者添加索引builder.HasIndex(x => x.SN).IsUnique();//序列號(hào)添加唯一索引builder.HasOne(r => r.User).WithMany(x=>x.Books)    .HasForeignKey(r => r.UserId).IsRequired();//導(dǎo)航屬性,本質(zhì)就是創(chuàng)建外鍵,雖然查詢很方便,生產(chǎn)中不建議使用!!!    }}

3.DBContext中應(yīng)用配置

protected override void OnModelCreating(ModelBuilder modelBuilder){    modelBuilder.HasCharSet("utf8mb4 ");    var (Assembly, Types) = _efEntitysInfo.EFEntitiesInfo;    foreach (var entityType in Types)    {modelBuilder.Entity(entityType);    }    //只需要將配置類所在的程序集給到,它會(huì)自動(dòng)加載    modelBuilder.ApplyConfigurationsFromAssembly(Assembly);    base.OnModelCreating(modelBuilder);}

Repository(倉(cāng)儲(chǔ))

這個(gè)不過(guò)分介紹,特別是基于http的微服務(wù)中基本都有這個(gè)。

1.創(chuàng)建一個(gè)倉(cāng)儲(chǔ)基類,對(duì)于不同的實(shí)體,創(chuàng)建一樣的增刪改查方法。

簡(jiǎn)單寫幾個(gè)查詢的方法定義。

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.創(chuàng)建倉(cāng)儲(chǔ)實(shí)現(xiàn)類,將DBContext注入到構(gòu)造中

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的構(gòu)造方法中,并且注入Repository

public class EFCoreEleganceUseEFCoreModule : Module{    protected override void Load(ContainerBuilder builder)    {base.Load(builder);builder.RegisterModule<EFCoreEleganceUseDomainModule>(); //注入domain模塊builder.RegisterGeneric(typeof(GenericRepository<,>))//將dbcontext注入到倉(cāng)儲(chǔ)的構(gòu)造中.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();    }}

數(shù)據(jù)庫(kù)配置

1.注入DBContext,從配置文件讀取數(shù)據(jù)庫(kù)配置,然后根據(jù)開(kāi)發(fā)/生產(chǎn)環(huán)境做一些特殊處理

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);    //開(kāi)發(fā)環(huán)境可以打開(kāi)日志記錄和顯示詳細(xì)的錯(cuò)誤    if (hostContext.HostingEnvironment.IsDevelopment())    {options.EnableSensitiveDataLogging();options.EnableDetailedErrors();    }});

項(xiàng)目架構(gòu)和源碼

項(xiàng)目只是一個(gè)demo架構(gòu),并不適用于生產(chǎn),主程序是一個(gè)控制臺(tái)項(xiàng)目,只需要引用相關(guān)的包和模塊,就可以啟動(dòng)一個(gè)web host.

全部代碼已經(jīng)全部上傳到github:https://github.com/BruceQiu1996/EFCoreDemo該項(xiàng)目是一個(gè)可以啟動(dòng)運(yùn)行的基于.net6的控制臺(tái)項(xiàng)目,啟動(dòng)后會(huì)啟動(dòng)一個(gè)web host和一個(gè)swagger頁(yè)面。

以上就是.net如何優(yōu)雅的使用EFCore實(shí)例詳解的詳細(xì)內(nèi)容,更多關(guān)于.net使用EFCore的資料請(qǐng)關(guān)注其它相關(guān)文章!

標(biāo)簽: ASP.NET
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
国产欧美日韩| 欧美大尺度电影在线| 国产69精品一区二区亚洲孕妇| 在线视频国内自拍亚洲视频| 亚洲免费观看高清完整版在线| 99久久婷婷国产| 精品乱码亚洲一区二区不卡| 风流少妇一区二区| 91精品国产高清一区二区三区蜜臀| 蜜桃视频一区二区| 欧美三级午夜理伦三级中视频| 日韩电影网1区2区| 欧洲视频一区二区| 毛片av一区二区三区| 欧美又粗又大又爽| 麻豆一区二区99久久久久| 在线视频国内一区二区| 精品影视av免费| 欧美日本高清视频在线观看| 久久狠狠亚洲综合| 制服丝袜激情欧洲亚洲| 国产成人av一区二区三区在线 | 制服丝袜激情欧洲亚洲| 国产精品18久久久久久久久| 日韩欧美123| 91麻豆免费在线观看| 国产精品麻豆视频| 国产三区二区一区久久| 午夜久久久影院| 色嗨嗨av一区二区三区| 久久成人久久鬼色| 欧美一级片在线观看| 成人手机电影网| 精品理论电影在线观看| 欧美成人亚洲| 国产精品久久久久aaaa| 亚洲免费激情| 丝袜美腿成人在线| 欧美日韩一级视频| 成人免费视频一区| 国产欧美精品区一区二区三区| 欧美午夜精品理论片a级大开眼界| 国产精品久久久久毛片软件| 亚洲大片在线| 婷婷久久综合九色国产成人 | 欧美激情一区二区三区在线视频| 中文幕一区二区三区久久蜜桃| 亚洲国产专区校园欧美| 亚洲国产视频在线| 欧美无砖砖区免费| 成人的网站免费观看| 国产精品不卡一区二区三区| 性娇小13――14欧美| 精品一区二区免费看| 26uuu精品一区二区| 亚洲国产一区二区三区高清| 日本vs亚洲vs韩国一区三区二区| 欧美一区国产二区| 欧美视频不卡| 无码av免费一区二区三区试看 | 91丨porny丨蝌蚪视频| 国产精品乱人伦| 久久九九电影| 国产经典欧美精品| 国产精品黄色在线观看| 久久一二三四| 成人a级免费电影| 日韩理论在线观看| 欧美色精品在线视频| 99久久亚洲一区二区三区青草| 亚洲国产精品v| 麻豆久久精品| 国产99久久久精品| 国产精品久久久久久户外露出| 久久久久久久久久久一区| 国产成人精品三级| 亚洲日穴在线视频| 精品视频在线看| 欧美日韩免费| 日本最新不卡在线| 国产午夜精品一区二区三区四区| 国产一区二区三区免费不卡 | 好看不卡的中文字幕| 午夜影视日本亚洲欧洲精品| 3751色影院一区二区三区| 伊人久久av导航| 激情综合色丁香一区二区| 国产午夜精品一区二区 | 色先锋资源久久综合| 成人激情视频网站| 亚洲一区二区精品久久av| 日韩免费电影一区| 亚洲欧美精品| 91麻豆福利精品推荐| 日韩影院在线观看| 国产亚洲美州欧州综合国| 香蕉av777xxx色综合一区| 从欧美一区二区三区| 亚洲一区二区三区中文字幕| 欧美一级日韩一级| 国产精品最新自拍| 91性感美女视频| 蜜臀91精品一区二区三区| 国产欧美日韩中文久久| 欧洲另类一二三四区| 亚洲高清123| 成人综合婷婷国产精品久久| 亚洲第一福利一区| 欧美激情在线一区二区三区| 欧美日韩精品系列| 亚洲一区二区成人| kk眼镜猥琐国模调教系列一区二区| 亚洲午夜精品网| 欧美激情一区二区三区全黄| 欧美精品色综合| 亚洲综合国产激情另类一区| 欧美一区二视频在线免费观看| 久久精品国产网站| 夜色激情一区二区| 国产农村妇女毛片精品久久麻豆| 欧美浪妇xxxx高跟鞋交| 午夜亚洲性色福利视频| 欧美激情无毛| 懂色av中文一区二区三区| 日韩av网站在线观看| 亚洲人成7777| 久久综合色8888| 欧美精品三级日韩久久| 久久精品成人一区二区三区蜜臀| 欧美精品一级| 成人亚洲精品久久久久软件| 美日韩一区二区| 一区二区三区不卡视频| 国产欧美日韩另类视频免费观看| 欧美一区二区三区视频免费播放| 色视频成人在线观看免| 国产欧美日韩一级| 欧美日本在线| 91原创在线视频| 国产黑丝在线一区二区三区| 久久精品99国产精品日本| 视频在线观看91| 综合久久国产九一剧情麻豆| 久久免费电影网| 欧美大度的电影原声| 欧美日本视频在线| 在线视频国产一区| 噜噜噜躁狠狠躁狠狠精品视频| 伊甸园精品99久久久久久| 91亚洲国产成人精品一区二区三| 国产不卡高清在线观看视频| 国产又粗又猛又爽又黄91精品| 日本不卡一二三| 婷婷丁香久久五月婷婷| 一区二区成人在线观看| 亚洲女女做受ⅹxx高潮| 国产精品福利一区| 久久九九久久九九| 久久九九全国免费| 国产视频在线观看一区二区三区 | 日韩三级电影网址| 欧美日韩成人高清| 欧美色视频在线| 在线观看亚洲精品| 欧美在线综合视频| 色94色欧美sute亚洲线路二| 久久婷婷一区| 久久久久久九九九九| 亚洲一区二区在| 亚洲专区一区二区三区| 亚洲一区二区三区四区五区午夜| 99人久久精品视频最新地址| 中日韩男男gay无套| 国产日韩一区二区三区在线播放| 夜久久久久久| 亚洲一区二区三区涩| 久久高清国产| 一本色道**综合亚洲精品蜜桃冫| 麻豆久久久9性大片| 久久永久免费| 欧美中文字幕不卡| 3d成人h动漫网站入口| 日韩你懂的在线观看| 2020国产精品久久精品美国| 国产视频一区不卡| 国产精品无遮挡| 亚洲精品日韩综合观看成人91| 一区二区久久久久久| 日韩精品一二三| 精品亚洲porn| 成人理论电影网| 欧美一区二区三区在线播放| 欧美日本中文| 国产欧美日韩综合一区在线观看 | 国产三级一区二区三区| 国产精品伦理在线| 亚洲精品国产一区二区精华液| 亚洲成人av一区| 久久精品99国产精品日本| 国产成人精品亚洲日本在线桃色|