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);
        }
    }
}

ASP.. NET MVC 5页面挂在数据库调用上

该错误是因为您正在控制器中实例化dbContext对象,而不是从自己的上下文中获取它。存在两个不同的上下文对象,一个在控制器中,一个在启动中,在同一个数据库上工作,导致了这个死锁。解决这个问题最简单的方法是替换代码

            db = new ApplicationDbContext();

db = System.Web.HttpContext.Current.GetOwinContext().Get<ApplicationDbContext>();

修复

我有同样的问题,这就是我如何解决它。

  1. 删除当前数据库(我使用的是。mdf)
  2. 创建新数据库
  3. 清理和重建
  4. 更新数据库-如果需要-
遗憾的是,我不知道问题的根源。如果你知道,请编辑答案/发表评论:)

对于有相同问题的其他人,请确保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指出这一点,因为它极大地拯救了我。我本想给你的回答点赞,但不幸的是,我还没有足够的声誉:)