【问题标题】:Two dbcontext for the same database同一个数据库的两个 dbcontext
【发布时间】:2019-04-15 19:33:02
【问题描述】:

我有以下两个项目的解决方案:

  • Razor 类库
  • ASP.NET Core Web 应用程序

在类库中,我有一个这样的 dbcontext:

public partial class ApplicationDbContext : IdentityDbContext<ApplicationUser, ApplicationRole, string>
{
    public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options)
        : base(options)
    {
    }

    public DbSet<Customer> Customers { get; set; }
}

在 Web 应用程序中,我使用 Startup.cs 文件中的以下行将 dbcontext 注册为服务

        services.AddDbContext<ApplicationDbContext>(options =>
            options.UseSqlServer(
                Configuration.GetConnectionString("ApplicationDbContextConnection")));

此时我可以在两个项目中使用 dbcontext。现在,在不更改类库的情况下,我想为同一个数据库定义新表,因此我在 Web 应用程序项目中创建了一个新的 dbcontext,它继承自类库中的那个。

public class ExtendedApplicationDbContext : ApplicationDbContext
{
    public ExtendedApplicationDbContext(DbContextOptions<ApplicationDbContext> options) : base(options)
    {
    }
    public DbSet<Product> Products { get; set; }
}

我更新了我的 Startup.cs 文件,以便同时注册新的 dbcontext,以便能够在 Web 应用程序中使用 Products 表。

        services.AddDbContext<ApplicationDbContext>(options =>
            options.UseSqlServer(
                Configuration.GetConnectionString("ApplicationDbContextConnection")));
        services.AddDbContext<ExtendedApplicationDbContext>(options =>
            options.UseSqlServer(
                Configuration.GetConnectionString("ApplicationDbContextConnection")));

现在我可以在两个项目中使用 ApplicationDbContext,但是当我尝试在 Web 应用程序中使用 ExtendedApplicationDbContext 时,发生了一些奇怪的事情。

我可以读取数据,但我所做的所有更改都不会更新数据库。

我试图解释它,但我找不到解决问题的方法。 有人可以帮我理解我的代码有什么问题吗?

【问题讨论】:

  • 你在正确的上下文中调用SaveChanges / await SaveChangesAsync 吗?你能显示你进行更改的代码吗?
  • 我使用 SaveChanges() 和 SaveChangesAsync() 的方式与我对它工作的 dbcontext 的使用方式相同。

标签: asp.net-core entity-framework-core


【解决方案1】:

EF 上下文保持对象缓存和内部更改跟踪。有两个独立的上下文引用同一个数据库将不可避免地导致同步问题,这将导致一个上下文破坏另一个上下文正在执行的操作,反之亦然。简而言之,每个数据库应该只有一个 active 上下文。

我指定 active 是因为可以使用“有界”上下文,在这种情况下,您会有多个实际上连接到同一个数据库的上下文,但这样的上下文应该只处理一个子集数据库中的对象,没有重叠。例如,您可能有一个与客户合作的上下文和一个与产品合作的上下文,但两者之间不应有交叉,即“产品”上下文不应以任何方式(即使只是关系)引用客户,反之亦然。

长短,这不是正确的路径。一方面,RCL 和 Web 应用程序都不应该有上下文。它应该位于不同的类库中,然后可以在其他两个项目之间共享。此外,Web 应用程序层不应对数据层进行任何架构更改,因为该数据层正在被其他事物使用。这是在脚上射击自己的好方法。把数据层的东西和数据层放在一起。

【讨论】:

  • 有没有办法只注册扩展的 dbcontext 并且我会以某种方式使用接口或泛型变量使其可用于类库?
  • 我用接口实现了,它可以工作。谢谢你,因为你帮助我找到了解决它的方法。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-08-05
  • 1970-01-01
  • 2018-04-07
  • 2017-05-05
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多