到目前为止,我的印象是DbContext是用来表示数据库的,因此,如果您的应用程序使用一个数据库,那么您只需要一个DbContext。
然而,一些同事希望将功能区域分解为单独的DbContext类。
我相信这是出于一个好的原因——希望保持代码更干净——但它似乎不稳定。我的直觉告诉我这是个坏主意,但不幸的是,我的直觉并不是设计决策的充分条件。
所以我在寻找:
A)为什么这可能是一个坏主意的具体例子;
B)保证这一切都会顺利解决。
到目前为止,我的印象是DbContext是用来表示数据库的,因此,如果您的应用程序使用一个数据库,那么您只需要一个DbContext。
然而,一些同事希望将功能区域分解为单独的DbContext类。
我相信这是出于一个好的原因——希望保持代码更干净——但它似乎不稳定。我的直觉告诉我这是个坏主意,但不幸的是,我的直觉并不是设计决策的充分条件。
所以我在寻找:
A)为什么这可能是一个坏主意的具体例子;
B)保证这一切都会顺利解决。
当前回答
提醒:如果你要组合多个上下文,请确保你在各种RealContexts.OnModelCreating()中剪切和粘贴所有功能到你的单个CombinedContext.OnModelCreating()。
我只是浪费时间寻找为什么我的级联删除关系没有被保存,却发现我没有移植modelBuilder.Entity<T>()....WillCascadeOnDelete();代码从我的真实上下文到我的组合上下文。
其他回答
提醒:如果你要组合多个上下文,请确保你在各种RealContexts.OnModelCreating()中剪切和粘贴所有功能到你的单个CombinedContext.OnModelCreating()。
我只是浪费时间寻找为什么我的级联删除关系没有被保存,却发现我没有移植modelBuilder.Entity<T>()....WillCascadeOnDelete();代码从我的真实上下文到我的组合上下文。
嗯,花了相当多的时间在每个DB模式的独立DB上下文的问题上,希望它能帮助其他人…
我最近开始从事一个项目,其中一个数据库有3个模式(DB优先方法),其中一个用于用户管理。每个单独的模式都有一个DB上下文。当然,用户也与其他模式相关,例如。schema KB有一个表Topic,它有“由谁创建的”,“最后修改的是谁”等。FK到标识模式,表appuser。
这些对象是在c#中单独加载的,首先,topic是从一个上下文加载的,然后用户是通过用户id从另一个db上下文加载的-不太好,必须修复这个! (类似于使用EF 6在同一个数据库中使用多个dbcontext)
首先,我尝试将身份模式中缺失的FK指令添加到KB模式中,添加到KB DB上下文中的EF modelBuilder中。就像只有1个上下文一样,但我把它分离成2个。
modelBuilder.Entity<Topic>(entity =>
{
entity.HasOne(d => d.Creator)
.WithMany(p => p.TopicCreator)
.HasForeignKey(d => d.CreatorId)
.HasConstraintName("fk_topic_app_users");
它没有工作,因为kb db context没有任何关于用户对象的信息,postgres返回错误关系“AppUsers”不存在。选择语句没有正确的模式,字段名等信息。
我几乎放弃了,但后来我注意到一个开关“-d”运行dotnet ef dbcontext脚手架。它是-data-annotations的缩写-使用属性配置模型(在可能的情况下)。如果省略,则只使用fluent API。 通过指定这个开关,对象属性不是在db上下文OnModelCreating()中定义的,而是在对象本身上定义的,带有属性。
通过这种方式,EF获得了足够的信息,可以用正确的字段名和模式生成正确的SQL语句。
TL;DR:单独的DB上下文不能很好地处理它们之间的关系(FKs),每个上下文只有关于自己实体的信息。 当在dotnet ef dbcontext scaffold上指定“-data-annotations”开关时,这些信息不是存储在每个单独的上下文中,而是存储在DB对象本身上。
另一点“智慧”。我有一个面向互联网和内部应用程序的数据库。每一个面向都有一个上下文。这有助于我保持纪律严明的隔离。
首先,在代码中,可以有多个DBContext和一个数据库。您只需在构造函数中指定连接字符串。
public class MovieDBContext : DbContext
{
public MovieDBContext()
: base("DefaultConnection")
{
}
public DbSet<Movie> Movies { get; set; }
}
灵感来自@JulieLerman的DDD MSDN Mag文章2013
public class ShippingContext : BaseContext<ShippingContext>
{
public DbSet<Shipment> Shipments { get; set; }
public DbSet<Shipper> Shippers { get; set; }
public DbSet<OrderShippingDetail> Order { get; set; } //Orders table
public DbSet<ItemToBeShipped> ItemsToBeShipped { get; set; }
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Ignore<LineItem>();
modelBuilder.Ignore<Order>();
modelBuilder.Configurations.Add(new ShippingAddressMap());
}
}
public class BaseContext<TContext>
DbContext where TContext : DbContext
{
static BaseContext()
{
Database.SetInitializer<TContext>(null);
}
protected BaseContext() : base("DPSalesDatabase")
{}
}
如果你正在进行新的开发,你想让Code First基于你的类创建或迁移你的数据库,你将需要使用DbContext创建一个“超级模型”,其中包括构建一个表示数据库的完整模型所需的所有类和关系。但是,这个上下文不能继承自BaseContext。莱托