EF--EntityState相互轉換

来源:https://www.cnblogs.com/menglin2010/archive/2020/01/30/12240027.html
-Advertisement-
Play Games

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();                    
  }
}

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 環境變數是在操作系統中一個具有特定名字的對象, 它包含了一個或者多個應用程式所將使用到的信息。 Path是一個常見的環境變數,它告訴操作系統,當要求系統運行一個程式而沒有告訴它程式所在的完整路徑時,系統除了在當前目錄下尋找此程式外,還應到哪些目錄下尋找。 ·設置Path環境變數的步驟如下 1. 右鍵 ...
  • ·下載JDK 1. 下載地址,點擊進入: www.oracle.com/technetwork/java/javase/downloads/index.html 點擊下載JDK,會出現圖1-9所示的下載列表界面。首先,點擊”Accept License Agreement”,然後選擇對應的版本,下載 ...
  • JVM(Java Virtual Machine)就是一個虛擬的用於執行bytecode位元組碼的”虛擬電腦”。他也定義了指令集、寄存器集、結構棧、垃圾收集堆、記憶體區域。JVM負責將Java位元組碼解釋運行,邊解釋邊運行,這樣,速度就會受到一定的影響。 不同的操作系統有不同的虛擬機。Java 虛擬機機 ...
  • 電腦高級語言的類型主要有編譯型(如:C和C++)和解釋型(如:PHP、Python)兩種,而Java 語言是兩種類型的結合。 Java首先利用文本編輯器編寫 Java源程式,源文件的尾碼名為.java;再利用編譯器(javac)將源程式編譯成位元組碼文件,位元組碼文件的尾碼名為.class; 最後利用 ...
  • 這是Serilog系列的第三篇文章。 1. "第1部分 使用Serilog RequestLogging減少日誌詳細程度" 2. "第2部分 使用Serilog記錄所選的終結點屬性" 3. 第3部分 使用Serilog.AspNetCore記錄MVC屬性(本文) 4. 第4部分 從Serilog請 ...
  • WPF視窗充滿了各種元素,但這些元素中只有一部分是控制項。在WPF領域,控制項通常被描述為與用戶交互的元素——能接收焦點並接受鍵盤或滑鼠輸入的元素。明顯的例子包括文本框和按鈕。然而,這個區別有時有些模糊。將工具提示視為控制項,因為它根據用戶滑鼠的移動顯示或消失。將標簽視為控制項,因為它支持記憶碼(mnemo ...
  • 多點觸控(multi-touch)是通過觸摸屏幕與應用程式進行交互的一種方式。多點觸控輸入和更傳統的基於筆(pen-based)的輸入的區別是多點觸控識別手勢(gesture)——用戶可移動多根手指以執行常見操作的特殊方式。例如,在觸摸屏上放置兩根手指並同時移動他們,這通常意味著“放大",而以一根手 ...
  • " 返回《C 併發編程》" "1. 線程池的由來" "1.1. 線程池出現前" "1.2. 線程池的誕生" "1.3. CLR線程池工作過程" "2. 線程池解決的問題" "2.1. 非同步調用方法" "2.2. 按時間間隔調用方法" "3. 當單個內核對象接收到信號通知時調用方法" "3.1. 註冊 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...