我知道这个问题有很多重复的地方,但我找不到一个适合我的场景。
所以我使用ASP。. NET MVC 4 +实体框架+ Ninject使用存储库模式(我看到很多提到存储库+工作单元模式?这可能是一个潜在的解决我的问题,但我不知道如何实现它)。
当我尝试添加一个新的帖子时,我得到"一个实体对象不能被IEntityChangeTracker的多个实例引用"错误在下面的代码行
context.Posts.Add(post);
这是我的完整实现:
具体的存储库
public class EFBlogRepository : IBlogRepository
{
private readonly EFDbContext context;
public EFBlogRepository(EFDbContext dbcontext)
{
context = dbcontext;
}
//add post
public int AddPost(Post post)
{
context.Posts.Add(post);
context.SaveChanges();
return post.PostID;
}
public Category Category(int id)
{
return context.Categories.FirstOrDefault(c => c.CategoryID == id);
}
public Tag Tag(int id)
{
return context.Tags.FirstOrDefault(t => t.TagID == id);
}
}
界面
public interface IBlogRepository
{
int AddPost(Post post);
Category Category(int id);
Tag Tag(int id);
}
我的控制器
public class AdminController : Controller
{
private IBlogRepository repository;
public AdminController(IBlogRepository repo)
{
repository = repo;
}
[HttpPost]
public ContentResult AddPost(Post post)
{
string json;
ModelState.Clear();
if (TryValidateModel(post))
{
var id = repository.AddPost(post);
json = JsonConvert.SerializeObject(new
{
id = id,
success = true,
message = "Post added successfully."
});
}
else
{
json = JsonConvert.SerializeObject(new
{
id = 0,
success = false,
message = "Failed to add the post."
});
}
return Content(json, "application/json");
}
}
我不认为上述任何一个都是问题的根源,我认为问题出在我的自定义模型绑定器
public class PostModelBinder : DefaultModelBinder
{
public override object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
{
var post = (Post)base.BindModel(controllerContext, bindingContext);
var repository = DependencyResolver.Current.GetService<IBlogRepository>();
if (post.Category != null)
post.Category = repository.Category(post.Category.CategoryID);
var tags = bindingContext.ValueProvider.GetValue("Tags").AttemptedValue.Split(',');
if (tags.Length > 0)
{
post.Tags = new List<Tag>();
foreach (var tag in tags)
{
post.Tags.Add(repository.Tag(int.Parse(tag.Trim())));
}
}
return post;
}
}
和我的global.asax.cs
ModelBinders.Binders.Add(typeof(Post), new PostModelBinder());
这是我的Ninject依赖解析器
public class NinjectDependencyResolver: IDependencyResolver
{
private IKernel kernel;
public NinjectDependencyResolver()
{
kernel = new StandardKernel();
AddBindings();
}
public object GetService(Type serviceType)
{
return kernel.TryGet(serviceType);
}
public IEnumerable<object> GetServices(Type serviceType)
{
return kernel.GetAll(serviceType);
}
private void AddBindings()
{
kernel.Bind<IBlogRepository>().To<EFBlogRepository>();
kernel.Bind<IAuthProvider>().To<FormsAuthProvider>();
}
}
你应该在对象绑定中绑定你的上下文InRequestScope
kernel.Bind<EFDbContext >().To<EFDbContext >().InRequestScope();
如错误提示所示,一个实体不能绑定到多个EF上下文。似乎您正在从一个上下文中检索实体,然后将其添加到另一个上下文中。使用上面的行,你告诉Ninject在同一个HTTP请求中使用相同的上下文实例来服务所有依赖项。
正在创建两个存储库。一个在控制器IBlogRepository repository
,另一个在模型绑定器var repository = DependencyResolver.Current.GetService<IBlogRepository>()
。在修复之前,每个存储库都有一个上下文的新实例,从而导致错误。修复后,两个存储库将共享上下文的相同实例。