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
。
如果我说错了,或者不清楚,请发表评论。 :)
我有一个引用 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
。
如果我说错了,或者不清楚,请发表评论。 :)