数据库上下文工厂是一种用于创建和管理数据库上下文实例的设计模式,常见于使用 Entity Framework 的 C# 应用程序中。它的主要作用是将数据库上下文(DbContext)的创建过程集中化、标准化,便于在不同场景下(如依赖注入、多租户、测试等)灵活控制上下文的生命周期。
为什么需要数据库上下文工厂?
在实际开发中,直接 new 一个 DbContext 可能导致资源管理混乱、连接泄漏或与依赖注入容器不兼容。通过工厂模式,可以按需生成上下文实例,同时配合服务容器更好地管理其作用域。
IDbContextFactory 接口的使用
从 Entity Framework Core 5.0 开始,微软引入了 IDbContextFactory
使用步骤如下:
定义你的数据库上下文类,继承自 DbContext 注册工厂服务到依赖注入容器 在需要的地方通过工厂创建上下文实例代码示例
假设有一个简单的 BloggingContext:
<font face="Courier New">
public class BloggingContext : DbContext
{
public DbSet<Blog> Blogs { get; set; }
<pre class='brush:php;toolbar:false;'>protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
optionsBuilder.UseSqlServer("YourConnectionString");
}}
在 Program.cs 或启动类中注册工厂:
<font face="Courier New">
var builder = WebApplication.CreateBuilder(args);
<p>builder.Services.AddDbContextFactory<BloggingContext>(options =>
options.UseSqlServer("YourConnectionString"));</p><p>var app = builder.Build();
</font>在某个服务或页面模型中使用工厂创建上下文:
<font face="Courier New">
public class BlogService
{
private readonly IDbContextFactory<BloggingContext> _contextFactory;
<pre class='brush:php;toolbar:false;'>public BlogService(IDbContextFactory<BloggingContext> contextFactory)
{
_contextFactory = contextFactory;
}
public async Task AddBlogAsync(string url)
{
var context = _contextFactory.CreateDbContext();
var blog = new Blog { Url = url };
context.Blogs.Add(blog);
await context.SaveChangesAsync();
}}
适用场景
这种模式特别适合以下情况:
需要在后台任务中访问数据库(如 IHostedService) 多线程环境中避免共享上下文实例 单元测试时隔离数据操作 动态切换连接字符串的多租户应用基本上就这些。使用数据库上下文工厂能让上下文创建更可控,也更符合现代 .NET 应用的设计规范。
