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 생성
더 깊이 공부하고 싶다면
DevLec에서는 실무 중심의 C#, .NET, ASP.NET Core, Blazor, 데이터 액세스 강좌를 단계별로 제공합니다. 현재 수강 가능한 강좌 외에도 더 많은 과정이 준비되어 있습니다.
DevLec.com에서 자세한 커리큘럼을 확인해 보세요.
DevLec 공식 강의
C# Programming
C# 프로그래밍 입문
프로그래밍을 처음 시작하는 입문자를 위한 C# 기본기 완성 과정입니다.
ASP.NET Core 10.0
ASP.NET Core 10.0 시작하기 MVC Fundamentals Part 1 MVC Fundamentals Part 2
웹 애플리케이션의 구조와 MVC 패턴을 ASP.NET Core로 실습하며 익힐 수 있습니다.
Blazor Server
풀스택 웹개발자 과정 Part 1 풀스택 웹개발자 과정 Part 2 풀스택 웹개발자 과정 Part 3
실무에서 바로 활용 가능한 Blazor Server 기반 관리자·포털 프로젝트를 만들어 봅니다.
Data & APIs
Entity Framework Core 시작하기 ADO.NET Fundamentals Blazor Server Fundamentals Minimal APIs
데이터 액세스와 Web API를 함께 이해하면 실무 .NET 백엔드 개발에 큰 도움이 됩니다.
VisualAcademy Docs의 모든 콘텐츠, 이미지, 동영상의 저작권은 박용준에게 있습니다. 저작권법에 의해 보호를 받는 저작물이므로 무단 전재와 복제를 금합니다. 사이트의 콘텐츠를 복제하여 블로그, 웹사이트 등에 게시할 수 없습니다. 단, 링크와 SNS 공유, Youtube 동영상 공유는 허용합니다. www.VisualAcademy.com
박용준 강사의 모든 동영상 강의는 데브렉에서 독점으로 제공됩니다. www.devlec.com