所以我开始在我的网站上使用EF(C#),我遇到了一些麻烦。用户可以创建或修改数据,他们将使用选择屏幕(第1页)执行此操作。如果用户选择创建新数据,我将执行以下代码:
Program newProg = new Program();
using (DatabaseEntities context = new DatabaseEntities())
{
Guid id = new Guid(list.SelectedValue);
var itemString = from item in context.Set where item.Id == id select item;
Item selectedItem = itemString.ToList()[0];
newProg.Items.Add(selectedItem);
context.AddToProgramSet(newProg);
context.Detach(newProg);
}
几乎创建了一个新的“程序”实例,它将沿着每个用户控件传递,直到用户准备好提交到数据库。此时将执行以下代码:
using (DatabaseEntities context = new DatabaseEntities())
{
context.AddToProgramSet(this.SummaryControl.SelectedProgram);
context.SaveChanges();
}
不幸的是,当我到达那里时,我收到以下消息:
无法将对象添加到ObjectStateManager,因为它已具有EntityKey。使用ObjectContext.Attach附加具有现有密钥的对象。
在这一行:
context.AddToProgramSet(this.SummaryControl.SelectedProgram);
另外,当我在前一行之前添加以下行时:
context.Attach(this.SummaryControl.SelectedProgram);
我收到此错误:
具有null EntityKey值的对象无法附加到对象上下文。
非常感谢任何帮助。感谢。
答案 0 :(得分:1)
此错误的根本原因是您尝试将实体添加为已设置其主键的新实体。
您在什么时候将实体添加到SummaryControl?您的第一个代码段显示您添加了实体:
...
newProg.Items.Add(selectedItem);
context.AddToProgramSet(newProg);
context.Detach(newProg);
...
然后你似乎再次添加它:
using (DatabaseEntities context = new DatabaseEntities())
{
context.AddToProgramSet(this.SummaryControl.SelectedProgram);
context.SaveChanges();
}
如果newProg
和this.SummaryControl.SelectedProgram
是同一个实体,则您尝试将其添加两次。从第一个代码段中删除context.AddToProgramSet(newProg);
,您是否在Entity上工作,然后将其添加到ProgramSet。
答案 1 :(得分:0)
Dave是对的,根本原因是实体已经拥有主键。
您在代码中未显示的一件事是:
Program newProg = new Program();
变为:
this.SummaryControl.SelectedProgram
您可以解决的一种方法是在开始向其添加项目之前将newProg保存到数据库。