我刚开始使用ASP.net 5,MVC6和Entity Framework 7开展实验项目。我的ASP.Net Identity工作正常,但后来尝试将我自己的一些数据添加到DbContext并点击此问题。 EF7报道:
处理请求时发生未处理的异常。
InvalidOperationException:未配置任何数据库提供程序。 通过覆盖您的OnConfiguring来配置数据库提供程序 DbContext类或在设置时的AddDbContext方法 服务。
Microsoft.Data.Entity.Internal.DatabaseProviderSelector.SelectServices(ServiceProviderSource providerSource)堆栈查询Cookie标头
InvalidOperationException:未配置任何数据库提供程序。 通过覆盖您的OnConfiguring来配置数据库提供程序 DbContext类或在设置时的AddDbContext方法 服务。
这是我的配置方法:
public void ConfigureServices(IServiceCollection services)
{
// Add framework services.
services.AddApplicationInsightsTelemetry(Configuration);
services.AddEntityFramework()
.AddSqlServer()
.AddDbContext<ApplicationDbContext>(options =>
options.UseSqlServer(Configuration["Data:DefaultConnection:ConnectionString"]));
services.AddIdentity<ApplicationUser, IdentityRole>()
.AddEntityFrameworkStores<ApplicationDbContext>()
.AddDefaultTokenProviders();
services.AddMvc();
// Add application services.
services.AddTransient<IEmailSender, AuthMessageSender>();
services.AddTransient<ISmsSender, AuthMessageSender>();
services.AddTransient<ApplicationUnitOfWork>(instance => new ApplicationUnitOfWork());
}
ApplicationUnitOfWork
是EF7上的一个外观,可以减少紧耦合。到目前为止,这是整个事情:
public class ApplicationUnitOfWork : IUnitOfWork
{
readonly ApplicationDbContext dbContext;
public ApplicationUnitOfWork()
{
dbContext = new ApplicationDbContext();
Products = new ProductRepository(dbContext);
}
public void Dispose() { dbContext.Dispose(); }
public IUserRepository Users { get; }
public IRepository<SoftwareProduct> Products { get; }
public async Task CommitAsync() { await dbContext.SaveChangesAsync(); }
public void Cancel() { throw new NotImplementedException(); }
}
现在,当我运行Web应用程序时,我可以注册用户帐户并登录,EF可以创建数据库,创建Identity表并使用用户数据填充它们。还创建了products表 - 所以在某种程度上,EF可以使用ApplicationDbContext
并找到它用来创建数据库和模式的提供程序。
但是,当我尝试访问使用ProductsRepository
的产品控制器时,EF会抱怨 - 即使在两种情况下都使用了ApplicationDbContext
并且用于创建数据库!
那怎么来的? ASP.net Identity有什么特别的,它可以某种方式获得提供者,但我自己的代码不能?我错过了什么“神奇的咒语”?
答案 0 :(得分:6)
因为你正在新建它,而不是为你注射它。你正在新建的那个还没有配置好。 你应该改变你的类,让它传递给构造函数。
DbContext有多个构造函数,你使用的是没有构造函数的空构造函数。
最好通过制作这样的构造函数来注入它:
public ApplicationUnitOfWork(ApplicationDbContext context)
{
dbContext = context;
Products = new ProductRepository(dbContext);
}
你的代码显示ApplicationDbContext已经注册了DI,Identity正在使用注入的那个,但你不是因为你自己使用无参数构造函数新建了它
您还应注册您的ApplicationUnitOfWork,以便注入:
services.AddScoped<ApplicationUnitOfWork, ApplicationUnitOfWork>();