防止Entity Framework重复插入关联对象
Entity Framework在數據庫與對象映射上做了很多工作,除了將數據庫里的表映射成相應的對象以外,它還能夠自動處理表之間的外鍵關系,并且可以用導航屬性(Navigation Property)的方式在對象層面上表示這些關系。
一般來說,當你插入一個對象時,Entity Framework默認會自動將對象通過導航屬性關聯的對象也插入到數據庫里面去,大部分情況下,這是我們想要的結果。當然,如果關聯的對象已經存在于數據庫當中時,Entity Framework會避免重復插入對象。但問題是,這個檢查對象已經存在避免重復插入數據的功能,好像只在一個Context(環境)下有效,即下面的代碼是可以正常執行的:
??????????? using (var context = new TestContext())
??????????? {
??????????????? var milestone = new Milestone()
??????????????? {
??????????????????? Title = "測試里程碑",
??????????????????? StartDate = DateTime.Now,
??????????????????? DueDate = DateTime.Now + TimeSpan.FromDays(30)
??????????????? };
?
??????????????? context.Milestones.Add(milestone);
??????????????? context.SaveChanges();
?
??????????????? id = milestone.Id;
?
??????????????? var project = new Project()
??????????????? {
????????????????? ??Title = "測試項目",
??????????????????? StartDate = DateTime.Now,
??????????????????? DueDate = DateTime.Now + TimeSpan.FromDays(30),
??????????????????? Owner = "測試用戶"
??????????????? };
?
??????????????? project.Children.Add(milestone);
??????????????? context.Project.Add(project);
??????????????? context.SaveChanges();
??????????? }
?
而如果對象是跨Context(環境)的話,或者基于現有對象復制的對象(包括主鍵也復制的情況),這就會產生重復插入的問題,因為新復制的對象,Entity Framework沒有辦法跟蹤對象的狀態,“誤以為”對象是一個全新的對象,比如,下面這段代碼就會導致Entity Framework拋出一個異常,異常根據Entity對象的數據庫約束不同,可能會報告不同的錯誤信息—這個問題一開始讓我迷惑了好幾天:
??????? public static void Main(string[] args)
??????? {
??????????? int id = 0;
??????????? using (var context = new TestContext())
??????????? {
??????????????? var milestone = new Milestone()
??????????????? {
??????????????????? Title = "測試里程碑",
??????????????????? StartDate = DateTime.Now,
??????????????????? DueDate = DateTime.Now + TimeSpan.FromDays(30)
??????????????? };
?
??????????????? context.Milestones.Add(milestone);
??????????????? context.SaveChanges();
?
??????????????? id = milestone.Id;
??????????? }
?
??????????? using ( var context = new TestContext())
??????????? {
??????????????? var project = new Project()
??????????????? {
??????????????????? Title = "測試項目",
??????????????????? StartDate = DateTime.Now,
??????????????????? DueDate = DateTime.Now + TimeSpan.FromDays(30),
??????????????????? Owner = "測試用戶"
??????????????? };
??????????????? var child = new Milestone() {
????? ??????????????Id = id
??????????????? };
?
??????????????? project.Children.Add(child);
???????????????????????????????
??????????????? context.Project.Add(project);
??????????????? context.SaveChanges();
??????????? }
??????? }
?
執行上面這段代碼,Entity Framework會在最后一個context.SaveChanges()上面拋出DbUpdateException,詳細信息是:“{"The conversion of a datetime2 data type to a datetime data type resulted in an out-of-range value.\r\nThe statement has been terminated."}”。這個異常一開始看上去太怪異了, 明明我將要保存的Project對象的所有DateTime類型都已經賦值(而且賦值都在范圍內)了,為什么還說超出賦值范圍呢?
后面才發現,這是因為,Entity Framework在插入project對象是,看到它的關聯對象列表Children里,有一個Milestone對象,而Milestone對象是重新復制的(只復制了ID)—這個場景是因為用戶在網頁上創建一個項目時,可以從里程碑列表里選擇一個事先創建好了的里程碑。由于Entity Framework沒有辦法跟蹤這個新復制的Milestone對象的狀態,所以它“誤認為”這個對象是一個新的對象,因此重新插入這個對象,而這個對象又沒有設置一些必要的日期屬性,導致了前面那個異常。
既然搞明白了道理,修復起來也很簡單,就是顯式告訴Entity Framework跟蹤這個對象—通過把第二個using段改成下面這樣:
??????????? using ( var context = new TestContext())
??????????? {
??????????????? var project = new Project()
??????????????? {
??????????????????? Title = "測試項目",
??????????????????? StartDate = DateTime.Now,
?????????????????? ?DueDate = DateTime.Now + TimeSpan.FromDays(30),
??????????????????? Owner = "測試用戶"
??????????????? };
??????????????? var child = new Milestone() {
??????????????????? Id = id
??????????????? };
?
??????????????? project.Children.Add(child);
????????????? ??var adapter = context as IObjectContextAdapter;
??????????????? adapter.ObjectContext.AttachTo("Milestones", child);
???????????????????????????????
??????????????? context.Project.Add(project);
??????????????? context.SaveChanges();
??????????? }
?
注意:我用的是Entity Framework CTP 5,采用的是代碼優先(code first)的方式創建的數據庫,但是本文提到的問題在數據庫優先和模型優先的情況里都是一樣的。
因為在網上找了好多文章都沒有提到這個問題,所以在這里記錄下來。
重現代碼:/Files/killmyday/codefirstef.zip
轉載于:https://www.cnblogs.com/killmyday/archive/2010/12/17/1909630.html
總結
以上是生活随笔為你收集整理的防止Entity Framework重复插入关联对象的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: dataGridView 行头那一块儿空
- 下一篇: wpf Command Binding