EF對數據做什麼樣的操作,是根據EF的上下文實體狀態決定,實體狀態有以下5種狀態,下麵我們就分別看下這5種狀態 數據準備,我們看到學生表裡有20000名學生記錄,最後1位學生的學生編號為0000020000 1、Detached--實體跟上下文壓根沒關係 我們看到我新創建了名學生,學號為000002 ...
- EF對數據做什麼樣的操作,是根據EF的上下文實體狀態決定,實體狀態有以下5種狀態,下麵我們就分別看下這5種狀態
數據準備,我們看到學生表裡有20000名學生記錄,最後1位學生的學生編號為0000020000
1、Detached--實體跟上下文壓根沒關係
我們看到我新創建了名學生,學號為0000020001,他是第20001位學生,此時列印上下文實體狀態,實體和上下文是分離狀態,沒有任何關係,Detached是表示連記憶體跟蹤都沒有建立,跟上下文dbContext沒有任何關係
Student newStudent = new Student() { Student_ID = "0000020001", Student_Name = "豬豬打屁股", Student_Sex = "男", Student_Identity_Card = "610102198307122319", Student_Birthday = DateTime.Now, Student_Email = "[email protected]", Student_Class = "[email protected]", Create_Time = DateTime.Now }; //1、Detached--實體跟context壓根沒關係 { using (SchoolDBEntities dbContext = new SchoolDBEntities()) { Console.WriteLine(dbContext.Entry<Student>(newStudent).State);//實體跟context沒關係 Detached newStudent.Student_Name = "小魚兒"; Console.WriteLine(dbContext.Entry<Student>(newStudent).State);//Detached dbContext.SaveChanges();//Detached啥事兒不發生 } }
2、Added--添加
我們看到剛開始,newStudent這個學生和上下文是Detached分離狀態,當把newStudent添加到上下文的Students集合里後,再列印newStudent的狀態,newStudent就是Added狀態了,當執行dbContext.SaveChange()後,把這個學生添加到表裡,最後列印newStudent的狀態,發現是Unchanged狀態,Unchanged的意思是上下文和newStudent建立了跟蹤,但是newStudent沒有發生改變
//2、Added--添加 { using (SchoolDBEntities dbContext = new SchoolDBEntities()) { Console.WriteLine(dbContext.Entry<Student>(newStudent).State);//Detached dbContext.Students.Add(newStudent);//插入數據(自增主鍵在插入成功後,會自動賦值過去) Console.WriteLine(dbContext.Entry<Student>(newStudent).State);//Added dbContext.SaveChanges(); Console.WriteLine(dbContext.Entry<Student>(newStudent).State);//Unchanged(跟蹤,但是沒變化) } }
3、Unchanged--跟蹤,但是沒變化
Unchanged就是上下文和實體建立了跟蹤,但是實體的值沒有發生改變
4、Modified--記憶體Clone
把上次添加的那個“豬豬打屁股”的學生查出來,列印實體狀態是Unchanged表示上下文建立了跟蹤,但是實體未改變,然後修改學生姓名為“豬大頭”,再打印實體狀態是Modified,表示實體已發生了修改,當執行dbContext.SaveChange()後,把對這個學生的修改保存到表裡,最後再列印newStudent的狀態,發現是Unchanged狀態,表示上下文和實體建立了跟蹤,但是實體的值沒有發生改變
{ using (SchoolDBEntities dbContext = new SchoolDBEntities()) { Student currentStudent = dbContext.Students.Find("0000020001");//即時查詢 Console.WriteLine(dbContext.Entry<Student>(currentStudent).State);//Unchanged(跟蹤,但是沒變化) currentStudent.Student_Name = "豬大頭";//修改--記憶體clone Console.WriteLine(dbContext.Entry<Student>(currentStudent).State);//Modified dbContext.SaveChanges();//更新資料庫,因為狀態是Modified
Console.WriteLine(dbContext.Entry<Student>(currentStudent).State);//Unchanged(跟蹤,但是沒變化)
}
}
5、Deleted--刪除
把上次修改的那個“豬大頭”的學生查出來,列印其實體狀態是Unchanged表示上下文建立了跟蹤,但是實體未改變,然後從上下文的Students集合中移除,再列印實體狀態是Deleted,表示實體已刪除,當執行dbContext.SaveChange()後,把這個學生從表裡刪除掉,最後再列印newStudent的狀態,發現是Detached狀態,表示上下文和實體已經分離,和上下文沒有任何關係了
{ using (SchoolDBEntities dbContext = new SchoolDBEntities()) { Student currentStudent = dbContext.Students.Find("0000020001");//即時查詢 Console.WriteLine(dbContext.Entry<Student>(currentStudent).State);//Unchanged(跟蹤,但是沒變化) dbContext.Students.Remove(currentStudent); Console.WriteLine(dbContext.Entry<Student>(currentStudent).State);//Deleted dbContext.SaveChanges();//刪除數據,因為狀態是Deleted Console.WriteLine(dbContext.Entry<Student>(currentStudent).State);//Detached已經從記憶體移除了 } }
- 實體和上下文建立跟蹤的兩種方式
1、查詢方式
我們看到通過主鍵學號為“0000020000”查詢出的這個學生實體和上下文建立了跟蹤,只不過學生實體值沒有發生任何改變,所以實體的狀態是Unchanged
{ using (SchoolDBEntities dbContext = new SchoolDBEntities()) { //1、查詢方式 Student currentStudent = dbContext.Students.Find("0000020000"); Console.WriteLine(dbContext.Entry<Student>(currentStudent).State); } }
2、Attach附加方式
修改前學號為0000020000的這名學生叫“石興江”,性別為“女”,new Student()1個學生只不過這個學生的學號是在Student表裡存在的“0000020000”,列印oldStudent實體狀態是Detached,跟dbContext壓根沒關係,然後把oldStudent附加到上下文中,再列印oldStudent實體狀態是Unchanged(這裡說明下:oldStudent的Student_Identity_Card,Student_Birthday,Student_Email等屬性是null,難道oldStudent實體狀態不應該是Modified嗎?難道不是把Student_Identity_Card,Student_Birthday,Student_Email改為null值嗎?這裡狀態還是Unchanged,說明瞭屬性值為null在EF里不是改為null,而是不做更改,最後一張圖證明瞭這點Student_Identity_Card,Student_Birthday,Student_Email等屬性在表裡沒有被改為null值),然後修改學號為0000020000的這名學生名稱為“石興江_Att”,性別為“男”,再列印oldStudent實體狀態是Modified(因為實體發生了變化),dbContext.SaveChanges();執行後,我們發現0000020000的這名學生名字被改為了“石興江_Att”,性別被改為了“男”,而Student_Identity_Card,Student_Birthday,Student_Email等欄位值沒有被改為null
{ using (SchoolDBEntities dbContext = new SchoolDBEntities()) { Student oldStudent = new Student() { Student_ID = "0000020000" }; Console.WriteLine(dbContext.Entry<Student>(oldStudent).State);//Detached--實體跟dbContext壓根沒關係 dbContext.Students.Attach(oldStudent); Console.WriteLine(dbContext.Entry<Student>(oldStudent).State);//Unchanged(跟蹤,但是沒變化) oldStudent.Student_Name = "石興江_Att"; oldStudent.Student_Sex = "男"; Console.WriteLine(dbContext.Entry<Student>(oldStudent).State);//Modified(因為實體發生了變化) dbContext.SaveChanges(); } }
- 按需更新,只更新指定的欄位
修改前,學號為“0000000001”的學生名字叫“趙峰真”,身份證號碼為"1234567890",我們先查詢出學號為“0000000001”的這名學生,列印實體狀態為Unchanged,表示實體建立了跟蹤但是實體未改變,然後修改學生姓名為“趙峰真001”,修改學生身份證號碼為“abc”,再列印實體狀態為Modified(因為實體發生了變化),然後指定這個學生的Student_Name屬性為已修改,指定Student_Identity_Card屬性為未被更改過,最後dbContext.SaveChanges();後,我們發現只有學生姓名變成了“趙峰真001” ,而學生身份號碼沒有被更改為abc“”
//按需更新--只修改指定的欄位 { using (SchoolDBEntities dbContext = new SchoolDBEntities()) { Student student = dbContext.Students.Find("0000000001");//即時查詢 Console.WriteLine(dbContext.Entry<Student>(student).State); student.Student_Name = "趙峰真001"; student.Student_Identity_Card = "abc"; Console.WriteLine(dbContext.Entry<Student>(student).State); dbContext.Entry<Student>(student).Property("Student_Name").IsModified = true;//指定欄位被改過 dbContext.Entry<Student>(student).Property("Student_Identity_Card").IsModified = false;//指定欄位未被改過 Console.WriteLine(dbContext.Entry<Student>(student).State); dbContext.SaveChanges(); } }