ASP.NET 将 DbContext 注入 Identity UserManager

ASP.NET Injecting DbContext into Identity UserManager

我有一个引用 AppUser 模型的问题模型。这是 1 到 * 的关系,因为 1 个 AppUser 有很多问题,一个问题属于 1 个 AppUser。我的问题 class 看起来像这样:

public class Question
{
    public int Id { get; set; }
    public string Subject { get; set; }
    public string Text { get; set; }
    public DateTime Date { get; set; }
    public int NumOfViews { get; set; }
    public AppUser LastAnswerBy { get; set; }

    public AppUser AppUser { get; set; }
    public ICollection<Comment> Comments { get; set; }
}

所以我尝试像这样在我的控制器中向数据库添加一个新问题:

[HttpPost]
    public ActionResult PostQuestion(Question question)
    {
        if (ModelState.IsValid)
        {
            var id = User.Identity.GetUserId();
            var user = UserManager.Users.FirstOrDefault(x => x.Id == id);
            question.Date = DateTime.Now;
            question.AppUser = user;

            _context.Questions.Add(question);
            _context.SaveChanges();
            return RedirectToAction("Index");
        }
        return View(question);
    }

    private AppUserManager UserManager
    {
        get { return HttpContext.GetOwinContext().GetUserManager<AppUserManager>(); }
    }

使用这段代码,我得到了这样的异常: 一个实体对象不能被 IEntityChangeTracker 的多个实例引用。所以在搜索了一下之后,问题似乎是我的控制器和 AppUserManager class 有 2 个不同的 DbContext 实例,解决方案是将它注入那些 classes。将它注入我的控制器没问题,但我不知道如何将它注入我的 UserManager class,看起来像这样:

public class AppUserManager : UserManager<AppUser>
{

    public AppUserManager(IUserStore<AppUser> store)
        : base(store)
    { }

    public static AppUserManager Create(IdentityFactoryOptions<AppUserManager> options,
    IOwinContext context)
    {
        AppIdentityDbContext db = context.Get<AppIdentityDbContext>();
        AppUserManager manager = new AppUserManager(new UserStore<AppUser>(db));

        return manager;
    }
}

它是从我的 IdentityConfig class 调用的,如下所示:

public class IdentityConfig
{
    public void Configuration(IAppBuilder app)
    {
        app.CreatePerOwinContext<AppIdentityDbContext>(AppIdentityDbContext.Create);
        app.CreatePerOwinContext<AppUserManager>(AppUserManager.Create);
        app.CreatePerOwinContext<AppRoleManager>(AppRoleManager.Create);

        app.UseCookieAuthentication(new CookieAuthenticationOptions
        {
            AuthenticationType = DefaultAuthenticationTypes.ApplicationCookie,
            LoginPath = new PathString("/Account/Login"),
        });

        app.UseExternalSignInCookie(DefaultAuthenticationTypes.ExternalCookie);
}

我的问题可能是我不太了解身份部分。非常感谢任何帮助

解法: 正如 Tobias 所说,我正在使用 2 个不同的上下文来更新数据库。像这样获取上下文有效:

private AppIdentityDbContext Context
    {
        get { return HttpContext.GetOwinContext().Get<AppIdentityDbContext>(); }
    }

你的问题很可能是你试图用两个不同的上下文做一件事。您正在使用一个上下文查找用户,并尝试使用另一个上下文更新数据库。要解决此问题,您应该像这样在控制器中实例化您的上下文:

_context = HttpContext.GetOwinContext().Get<AppIdentityDbContext>();

这样,您将获得相同的上下文,用于实例化您的 AppUserManager

如果我说错了,或者不清楚,请发表评论。 :)