Blazor Server에서 IDbContextFactory 사용하여 종속성 주입하기

  • 3 minutes to read

DbContext 팩터리 사용하기

Blazor Server에서 EF Core를 사용할 때에는 DbContext 개체를 직접 주입해서 사용하기보다는 IDbContextFactory로 주입 후 CreateDbContext 메서드를 호출하여 Context 개체를 생성하는 방법을 사용해야 합니다.

예를 들어, 동일 페이지를 여러 번 호출될 때 다음과 같은 문제를 경험할 수 있습니다.

[2022-12-13T02:15:05.052Z] Error: System.InvalidOperationException: A second operation was started on this context before a previous operation completed. This is usually caused by different threads concurrently using the same instance of DbContext.

이런 경우에는 DbContext 대신에 IDbContextFactory 인스턴스를 만들고 사용하면 문제가 해결됩니다.

다음 단계를 진행해서 DbContext 팩터리 개체를 만들고 사용할 수 있습니다.

ApplicationDbContext 클래스는 DbContextOptions<ApplicationDbContext> 매개 변수가 있는 public 생성자를 노출해야 합니다. 만약, 사용되지 않는 빈 생성자가 있으면 지우세요. 하나 이상의 생성자를 두지 마세요.

public class ApplicationDbContext : DbContext
{
    public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options)
        : base(options)
    {
    }
}

다음 샘플 코드는 Blazor Server 7.0 Fundamentals에서 사용한 DbContext 클래스입니다.

using Hawaso.Models.Candidates.CandidatesIncomes;
using Hawaso.Models.Candidates.CandidatesNames;
using Microsoft.EntityFrameworkCore;

namespace Hawaso.Models.Candidates
{
    public class CandidateAppDbContext : DbContext
    {
        public CandidateAppDbContext() : base()
        {
            // Empty
            // 만약, Repository 클래스에 생성자 매개 변수 주입 방식 사용시 이 생성자 제거 
        }

        public CandidateAppDbContext(DbContextOptions<CandidateAppDbContext> options) 
            : base(options)
        {
            // Empty
        }

        // DbSet of T 형태의 컬렉션 속성을 사용하여 모델(도메인)에 해당하는 테이블 생성
        public DbSet<Candidate> Candidates { get; set; } = null!;

        public DbSet<CandidateName> CandidatesNames { get; set; } = null!;
        
        public DbSet<CandidateIncome> CandidatesIncomes { get; set; } = null!;
    }
}

Program.cs 또는 Startup.cs(.NET 5 이전 버전)에서 AddDbContext 대신에 AddDbContextFactory 메서드를 호출합니다.

Startup.cs 파일의 일부

public void ConfigureServices(IServiceCollection services)
{
    services.AddDbContextFactory<ApplicationDbContext>(options =>
        options.UseSqlServer(@"Server=(localdb)\mssqllocaldb;Database=Test"));
}

Program.cs 파일의 일부

#region 새로운 DbContext 추가 - CandidateAppDbContext
// 새로운 DbContext 추가 

//[a] MVC, RazorPages, Web API에서는 DbContext 사용 가능
//builder.Services.AddDbContext<CandidateAppDbContext>(options => options.UseSqlServer(connectionString));

//[b] Blazor Server에서는 DbContextFactory 사용 권장
builder.Services.AddDbContextFactory<CandidateAppDbContext>(options => options.UseSqlServer(connectionString));
#endregion

그런 다음 생성자 주입을 통해 DbContextFactory 팩터리를 사용할 수 있습니다. 예를 들면 다음과 같습니다.

private readonly IDbContextFactory<ApplicationDbContext> _contextFactory;

public MyController(IDbContextFactory<ApplicationDbContext> contextFactory)
{
    _contextFactory = contextFactory;
}
@inject IDbContextFactory<CandidateAppDbContext> _ContextFactory

그러면 삽입된 팩터리를 사용하여 서비스 코드에서 DbContext 인스턴스를 생성할 수 있습니다. 예를 들면 다음과 같습니다.

public void DoSomething()
{
    using (var context = _contextFactory.CreateDbContext())
    {
        // ...
    }
}

다음 샘플 코드와 같이 using var 구문으로 더 줄여서 표현해도 됩니다.

// Blazor에서는 반드시 DbContext facotry 사용 권장(필수) 
// https://learn.microsoft.com/en-us/ef/core/dbcontext-configuration/
//Candidates = await _Context.Candidates.ToArrayAsync();
using var context = _ContextFactory.CreateDbContext(); // DbContext 생성
VisualAcademy Docs의 모든 콘텐츠, 이미지, 동영상의 저작권은 박용준에게 있습니다. 저작권법에 의해 보호를 받는 저작물이므로 무단 전재와 복제를 금합니다. 사이트의 콘텐츠를 복제하여 블로그, 웹사이트 등에 게시할 수 없습니다. 단, 링크와 SNS 공유, Youtube 동영상 공유는 허용합니다. www.VisualAcademy.com
박용준 강사의 모든 동영상 강의는 데브렉에서 독점으로 제공됩니다. www.devlec.com