1

我知道这个问题有很多重复,但我找不到适合我的情况的问题。

所以我正在使用 ASP.NET MVC 4 + Entity Framework + 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>();
    }
}
4

1 回答 1

2

您应该在 ninject 绑定中将上下文绑定为InRequestScope

kernel.Bind<EFDbContext >().To<EFDbContext >().InRequestScope();

正如错误所说,一个实体不能绑定到多个 EF 上下文。您似乎正在从一个上下文中检索实体,然后将其添加到另一个上下文中。使用上面的行,你告诉 Ninject 使用相同的上下文实例来服务于同一个 HTTP 请求中的所有依赖项。

正在创建两个存储库。一个在控制器IBlogRepository repository中,另一个在模型绑定器var repository = DependencyResolver.Current.GetService<IBlogRepository>()中。在修复之前,每个存储库都有一个新的上下文实例,从而导致错误。修复后,两个存储库将共享同一个上下文实例。

于 2014-08-13T15:35:45.203 回答