一个实体对象不能被多个IEntityChangeTracker错误实例引用



我知道这个问题有很多重复的地方,但我找不到一个适合我的场景。

所以我使用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>()。在修复之前,每个存储库都有一个上下文的新实例,从而导致错误。修复后,两个存储库将共享上下文的相同实例。

相关内容

  • 没有找到相关文章

最新更新