2017-09-14 169 views
5

我已经看到了一些关于这方面的类似例子,但我对这门语言还不够了解,只是还没有看到我做错了什么。我拼凑了一个演示来了解更多信息,但我在播种我的数据库时遇到了麻烦。ASP.NET核心2种子数据库

我收到以下错误:

InvalidOperationException: Cannot resolve scoped service 'demoApp.Models.AppDbContext' from root provider.

Microsoft.Extensions.DependencyInjection.ServiceLookup.CallSiteValidator.ValidateResolution(Type serviceType, ServiceProvider serviceProvider)

这里有三个文件有问题:

型号/ AppDbContext.cs

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

    } 
    public DbSet<Product> Products{ get; set; } 
    public DbSet<Category> Categories { get; set; } 
} 

型号/ DBInitializer.cs

public static class DbInitializer 
{ 
    public static void Seed(IApplicationBuilder applicationBuilder) 
    { 
     //I'm bombing here 
     AppDbContext context = applicationBuilder.ApplicationServices.GetRequiredService<AppDbContext>(); 

     if (!context.Products.Any()) 
     { 
      // Add range of products 
     } 

     context.SaveChanges(); 
    } 

    private static Dictionary<string, Category> _categories; 
    public static Dictionary<string, Category> Categories 
    { 
     get 
     { 
      if (_categories == null) 
      { 
       // Add categories... 
      } 

      return _categories; 
     } 
    } 
} 

Startup.cs

public Startup(IConfiguration configuration) 
{ 
    Configuration = configuration; 
} 

public IConfiguration Configuration { get; } 

public void ConfigureServices(IServiceCollection services) 
{ 
    services.AddTransient<ICategoryRepository, CategoryRepository>(); 
    services.AddTransient<IProductRepository, ProductRepository>(); 

    services.AddDbContext<AppDbContext>(options => 
     options.UseSqlServer(Configuration.GetConnectionString("DefaultConnection"))); 

    services.AddMvc(); 
} 

public void Configure(IApplicationBuilder app, IHostingEnvironment env) 
{ 
    if (env.IsDevelopment()) 
    { 
     app.UseDeveloperExceptionPage(); 
     app.UseBrowserLink(); 
     app.UseStatusCodePages(); 

     // Kersplat! 
     DbInitializer.Seed(app); 
    } 
    else ... 

    app.UseStaticFiles(); 
    app.UseMvc(routes => {...}); 
} 

有人能帮忙解释一下我做错了,如何补救?

回答

11

在ASP.NET Core 2.0中,建议进行以下更改。 (startup.cs中的种子适用于Core 1.x. For 2.0请进入Program.cs,修改Main方法以在应用程序启动时执行以下操作: 从依赖项注入容器中获取数据库上下文实例 调用种子方法,传递给它的上下文。 处置当种子法完成的上下文。 (下面是从微软网站的样本。https://docs.microsoft.com/en-us/aspnet/core/data/ef-mvc/intro

public static void Main(string[] args) 
{ 
var host = BuildWebHost(args); 

using (var scope = host.Services.CreateScope()) 
{ 
    var services = scope.ServiceProvider; 
    try 
    { 
     var context = services.GetRequiredService<yourDBContext>(); 
     DbInitializer.Seed(context);//<---Do your seeding here 
    } 
    catch (Exception ex) 
    { 
     var logger = services.GetRequiredService<ILogger<Program>>(); 
     logger.LogError(ex, "An error occurred while seeding the database."); 
    } 
} 

host.Run(); 
} 
+0

为什么修改'Main'?有没有提到这是最好的参考? –

+1

I t在上面的代码下面的链接文章中提到它。 “在较老的教程中,您可能会在Startup.cs中的Configure方法中看到类似的代码,我们建议您仅使用Configure方法来设置请求管道,应用程序启动代码属于Main方法。 – palehorse

+0

注意:为使CreateScope方法可用,需要使用以下using语句:'using Microsoft.Extensions.DependencyInjection;'。 – JohnLBevan

2

更新从原来的答案:

对于.NET 2.0的核心,看看this answer代替

原来的答案:

我不是.NET的核心专家任,但这可能是你的解决方案。

DBInitializer.cs

public static void Seed(IApplicationBuilder applicationBuilder) 
    { 
     using (var serviceScope = applicationBuilder.ApplicationServices.GetRequiredService<IServiceScopeFactory>() 
       .CreateScope()) 
     { 
      AppDbContext context = serviceScope.ServiceProvider.GetService<AppDbContext>(); 

      if (!context.Products.Any()) 
      { 
       // Seed Here 
      } 

      context.SaveChanges(); 
     } 
    } 

该错误表明上下文应当作用域。另外,如果你还没有这样做,我会看看Introduction to Dependency Injection in ASP.NET Core文档,但更具体地说,Service Lifetimes and Registration Options部分。

+0

我重新进行了迁移,更新数据库和数据库播种预期。我非常感谢你提供的链接。非常感谢你。 – forcequitIO