1

我有一个通用存储库:

    public class GenericRepository<TEntity> : AbstractRepository<TEntity>, IRepository<TEntity> where TEntity : class
    {
        private DbContext _context;
        [...]
        public GenericRepository(DbContext context)
        {
            _context = context;
            context.Configuration.AutoDetectChangesEnabled = true;
            _dbSet = _context.Set<TEntity>();
        }
        [...]
        public void SaveChanges()
        {
            _context.SaveChanges();
        }
        [...]
    public void Add(TEntity entity)
    {
        if (entity == null)
        {
            throw new ArgumentNullException("entity");
        }

        _dbSet.Add(entity);
    }
        [...]
    public virtual void Update(TEntity entity)
    {
        _context.Entry(entity).State = EntityState.Modified;
    }

在我的控制器中,有以下代码:

    [HttpPost]
    public ActionResult Edit(Project project)
    {
          if (ModelState.IsValid)
        {
            if (project.Id == 0)
            {
                ProjectRepository.Add(project);
            }
            else
            {
                ProjectRepository.Update(project);
            }

            ProjectRepository.SaveChanges();
            [...]

选择和插入工作正常,但更新失败:我收到 InvalidOperationException(德语错误消息的英文翻译是“对象状态管理器中已存在具有相同键的对象。对象状态管理器无法跟踪具有相同键的多个对象钥匙。”)。

我不明白,因为我是我的开发机器上唯一的用户,而且我没有在其他地方修改记录。

知道我在这里做错了什么吗?

4

3 回答 3

6

看看这些答案:

基本上你需要这样做:

 var entity = _context.Projects.Find(project.ProjectId);
 _context.Entry(entity).CurrentValues.SetValues(project);

希望这会有所帮助。

于 2012-07-25T11:03:06.967 回答
0

project实例是通过模型绑定创建的,而不是从您的存储库中加载的,因此您需要从存储库中实际加载一个project实例,然后更改其属性。

于 2012-07-25T11:12:26.290 回答
0

免责声明:我从未使用过实体框架,我是根据我在 ASP.NET MVC 和 nHibernate 方面的经验编写的,但您应该能够应用相同的模式

好的,首先您的实际问题实际上是双键插入,这是因为传递给您的编辑操作的项目对象与您尝试更新的对象不同(这个对象是从您的基于模型绑定器创建的FormValueProvider 中的值)。它可能具有 excat 相同的值,因此具有相同的 id,但对于您的 ORM,它是一个全新的对象,从未持久化到数据库中。

您可以通过使用以下模式来防止这种情况(快速的脏示例代码)

    [HttpPost]
    public ActionResult Edit(Project project)
    {
        if (ModelState.IsValid)
        {
            if (project.Id == 0)
            {
                ProjectRepository.Add(project);
            }
            else
            {
                Project dbProject = ProjectRepository.GetById(project.Id); //or whatever your method to retrieve entities by Id is named
                UpdateModel(dbProject, "project"); //http://msdn.microsoft.com/en-us/library/dd470933.aspx
                ProjectRepository.Update(project);
            }
        }
    }
于 2012-07-25T13:24:51.393 回答