ASP.NET核心2种子数据库
我已经看到了一些关于这方面的类似例子,但我对这门语言还不够了解,只是还没有看到我做错了什么。我拼凑了一个演示来了解更多信息,但我在播种我的数据库时遇到了麻烦。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 => {...});
}
有人能帮忙解释一下我做错了,如何补救?
在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();
}
更新从原来的答案:
对于.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部分。
我重新进行了迁移,更新数据库和数据库播种预期。我非常感谢你提供的链接。非常感谢你。 – forcequitIO
为什么修改'Main'?有没有提到这是最好的参考? –
I t在上面的代码下面的链接文章中提到它。 “在较老的教程中,您可能会在Startup.cs中的Configure方法中看到类似的代码,我们建议您仅使用Configure方法来设置请求管道,应用程序启动代码属于Main方法。 – palehorse
注意:为使CreateScope方法可用,需要使用以下using语句:'using Microsoft.Extensions.DependencyInjection;'。 – JohnLBevan