1

我有一个非常简单的模型:

public class Item : DbEntity
{
    public string Name {get;set;}
    public int Quantity {get;set;}
    public virtual Category Category {get;set;}
}

public class Category : DbEntity
{
    public string Name {get;set;}
}

我使用以下方法保存并提交对数据库的更改:

public void SaveEntity<TEntity>(TEntity entity) where TEntity : DbEntity
{
    if (entity.Id.Equals(Guid.Empty))
        _context.Set<TEntity>().Add(entity);
    else
        _context.Entry(entity).State = System.Data.EntityState.Modified;

    _context.SaveChanges();
}

public void DeleteEntity<TEntity>(TEntity entity) where TEntity : DbEntity
{
     _context.Set<TEntity>().Remove(entity);
     _context.SaveChanges();
}

这些保存和删除方法驻留在 Repository 中,这个 Repository 是 Ninjected,因此它应该在整个应用程序生命周期中保持恒定状态。这是一种久经考验的方法,它一直对我有用。现在由于某种原因,当我这样做时:

[HttpPost]
public ActionResult Edit(MenuItemViewModel vm)
{
    if (ModelState.IsValid)
    {
        // TODO: Add ViewModel Logic
        vm.MenuItem.Category = _repository.Categories.FirstOrDefault(x => x.Id.Equals(vm.SelectedCategory));

    try
    {
        // TODO: Add Save Logic
            _repository.SaveEntity(vm.MenuItem);

            TempData["success"] = true;
            TempData["message"] = String.Format("Saved {0} Successfully", vm.MenuItem.Description);
    }
    catch (Exception ex)
    {
             TempData["error"] = true;
             TempData["message"] = "Error Occured";
    }

    return _redirectTo;
    }

    return View("Shape", vm);
}

视图部分很简单,只有 and 的一些字段NameQuantityCategory是一个Drop Down包含所有类别名称 asText和 id as的框Value,它被发送回存储库所在的服务器,并收集新的Category并将其分配给实体。

出于某种原因,这只是没有对数据库进行任何更改,前后断点,当我在检查数据库后中断时,它没有改变!

任何人都可以看到阻止它更新的愚蠢的小错误。

编辑

刚刚阅读了这篇StackOverflow 帖子,我拿起了上面说的部分,你先附加它然后编辑它,因为当调用附加时,你用Unchanged状态调用它,然后当你修改它时,当你调用时更改将被保存SaveChanges().

跟这有关系吗?!

更新
好吧,一些有趣的消息,我逐步执行该SaveEntity方法,在我跳过SaveChanges()我再次检查上下文,它表明它已被更新?!我重定向到另一个控制器和操作的事实是否会影响结果或表修改?我不认为问题出在SaveEntity方法上!!

最近更新

我更改了的模型Item并简单地添加了public Guid CategoryId {get;set;}简单地更改Category为打开的行Cascade On Delete。现在,当我尝试时出现此错误SaveChanges

发生参照完整性约束冲突:定义参照约束的属性值在关系中的主体对象和从属对象之间不一致。

4

2 回答 2

1

所以,为了解决这个问题,我将Item实体更改为:

public class Item : DbEntity
{
    public string Name {get;set;}
    public int Quantity {get;set;}

    public Guid? CategoryId {get;set;}
    public virtual Category Category {get;set;}
}

这基本上使更改变得更容易和更快,Category因为您不必再​​次调用数据库。这?使它成为一个可以为空的属性,它设置Cascade DeleteNo Action这样当你删除Item它时它不会烦人地消失并删除关联的Categories.

我仍然不知道为什么原始方式不起作用,它在过去已经完成并且它曾经否定必须放置public Guid? CategoryId {get;set;},因为它会设置可为空的属性并且已经创建了一个Category_Id字段。但是嘿嗬,这就是它想要的样子,那我就去打球了!!

于 2013-06-06T13:10:13.013 回答
0

在 SaveEntity 中,如果 Id 不等于 Guid.Empty,则您没有将项目附加到上下文,只是将 State 设置为 Modified。尝试:

public void SaveEntity<TEntity>(TEntity entity) where TEntity : DbEntity
{
    if (entity.Id.Equals(Guid.Empty))
    {
        _context.Set<TEntity>().Add(entity); // Automatically sets state to Added
    }
    else 
    {
        _context.Set<TEntity>().Attach(entity);
        _context.Entry(entity).State = System.Data.EntityState.Modified;
    }

    _context.SaveChanges();
}

此外,对于 DeleteEntity,请尝试附加和删除:

public void DeleteEntity<TEntity>(TEntity entity) where TEntity : DbEntity
{
    _context.Set<TEntity>().Attach(entity); // Attaches entity with state Unchanged
    _context.Set<TEntity>().Remove(entity); // Changes entity state to Deleted
    _context.SaveChanges();
}

关键是在 ASP.NET 等断开连接的环境中,从帖子返回的模型不附加到上下文,因此需要重新附加才能处理它。

作为测试,在调用 SaveChanges 之前检查现有代码中实体的状态,它可能是 Unchanged 意味着不会对其采取任何操作。

于 2013-06-06T11:28:59.823 回答