ASP.. NET MVC 5页面挂在数据库调用上
本文关键字:数据库 调用 NET MVC 5页 ASP | 更新日期: 2023-09-27 18:17:57
我正在尝试使用ASP启动和运行MVC 5项目。NET Identity 2.0。我的出发点是本教程中的示例应用程序。我的初始页面是Home/Index。当我尝试做一个搜索(期望一个null返回值)的应用程序只是挂起,我不知道为什么。db上下文的实例化导致调用Seed()方法,这似乎很正常,但随后它挂在roleManager.FindByName(roleName)调用上(我在下面的代码中注释了它)。暂停调试器显示它被卡住的地方,但我不知道从那里去哪里。相关类如下:
控制器:
public class HomeController : ApplicationController
{
public ActionResult Index()
{
var user = db.Users.Find("dummyVal");
return View();
}
[Authorize]
public ActionResult About()
{
ViewBag.Message = "Your app description page.";
return View();
}
public ActionResult Contact()
{
ViewBag.Message = "Your contact page.";
return View();
}
}
基本控制器:
public abstract class ApplicationController : Controller
{
protected ApplicationDbContext db;
public ApplicationController()
{
db = new ApplicationDbContext();
}
}
DB初始化:
public class ApplicationDbInitializer : DropCreateDatabaseAlways<ApplicationDbContext>
{
protected override void Seed(ApplicationDbContext context) {
InitializeIdentityForEF(context);
base.Seed(context);
}
//Create User=Admin@Admin.com with password=Admin@123456 in the Admin role
public static void InitializeIdentityForEF(ApplicationDbContext db) {
var userManager = HttpContext.Current.GetOwinContext().GetUserManager<ApplicationUserManager>();
var roleManager = HttpContext.Current.GetOwinContext().Get<ApplicationRoleManager>();
const string name = "admin@example.com";
const string password = "Admin@123456";
const string roleName = "Admin";
//Create Role Admin if it does not exist
// EXECUTION HANGS ON THIS CALL...
var role = roleManager.FindByName(roleName);
if (role == null) {
role = new IdentityRole(roleName);
var roleresult = roleManager.Create(role);
}
var user = userManager.FindByName(name);
if (user == null) {
user = new ApplicationUser { UserName = name, Email = name };
var result = userManager.Create(user, password);
result = userManager.SetLockoutEnabled(user.Id, false);
}
// Add user admin to Role Admin if not already added
var rolesForUser = userManager.GetRoles(user.Id);
if (!rolesForUser.Contains(role.Name)) {
var result = userManager.AddToRole(user.Id, role.Name);
}
// Create dummy user
const string dummyUserName = "PeterVenkman";
const string dummyUserPwd = "gf%^^ftf83X";
var dummyUser = userManager.FindByName(dummyUserName);
if (dummyUser == null) {
dummyUser = new ApplicationUser { UserName = dummyUserName, Email = dummyUserName };
var result = userManager.Create(dummyUser, dummyUserPwd);
result = userManager.SetLockoutEnabled(dummyUser.Id, false);
}
}
}
该错误是因为您正在控制器中实例化dbContext对象,而不是从自己的上下文中获取它。存在两个不同的上下文对象,一个在控制器中,一个在启动中,在同一个数据库上工作,导致了这个死锁。解决这个问题最简单的方法是替换代码
db = new ApplicationDbContext();
db = System.Web.HttpContext.Current.GetOwinContext().Get<ApplicationDbContext>();
修复
我有同样的问题,这就是我如何解决它。
- 删除当前数据库(我使用的是。mdf)
- 创建新数据库
- 清理和重建
- 更新数据库-如果需要-
对于有相同问题的其他人,请确保ApplicationDbInitializer中的用户名/密码有效。例如,如果您将admin密码设置为123,则会挂起。
编辑:这篇文章提供了一个解释+答案http://forums.asp.net/post/5802779.aspx
Suhas Joshi有正确答案。ApplicationDbContext对象本质上应该是一个由"Microsoft asp.net Identity Owin"包(使用NuGet安装)管理的单例。当你在你的ApplicationController中手动创建它的一个新实例时,会出现对相同资源的争用,从而导致死锁。
请注意,使用的"Get()"扩展来自以下库,您必须在ApplicationContext类中引用该库。
using Microsoft.AspNet.Identity.Owin;
感谢Suhas Joshi指出这一点,因为它极大地拯救了我。我本想给你的回答点赞,但不幸的是,我还没有足够的声誉:)