日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

防止Entity Framework重复插入关联对象

發布時間:2024/4/17 编程问答 42 豆豆
生活随笔 收集整理的這篇文章主要介紹了 防止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重复插入关联对象的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。