EF6基本使用

来源:https://www.cnblogs.com/yuxl01/archive/2022/04/02/16084674.html
-Advertisement-
Play Games

1.EF基本搭建 EF也用了好幾年了,但是在日常開發的時候,有時候因為偷懶,有時候因為趕項目,很多代碼,多半就是Ctrl+C和Ctrl+V,慢慢的一些代碼怎麼寫都忘記了,雖然覺得很簡單,但是就是記不起來怎麼寫,逐漸退化,所以記錄一下,後續再賦值粘貼也好找一些,免得打開項目。 在此以.Net Fram ...


1.EF基本搭建

EF也用了好幾年了,但是在日常開發的時候,有時候因為偷懶,有時候因為趕項目,很多代碼,多半就是Ctrl+C和Ctrl+V,慢慢的一些代碼怎麼寫都忘記了,雖然覺得很簡單,但是就是記不起來怎麼寫,逐漸退化,所以記錄一下,後續再賦值粘貼也好找一些,免得打開項目。

在此以.Net FramWork 控制台搭建簡單的Demo使用的模式是DBFirst,個人覺得現在多半還是資料庫設計和代碼還是分開的。

1.創建資料庫和表,這裡使用關聯表,一個主表一個從表,抽象出簡單的業務關係為一個員工持有哪些設備,而關係是一對多。

--創建資料庫
CREATE DATABASE EfDemo;
GO
USE EfDemo;
GO
--創建員工表
CREATE TABLE [dbo].[Employee]
([id]   [INT] IDENTITY(1, 1) NOT NULL, 
 [Code] [NVARCHAR](20) NULL, 
 [Name] [NVARCHAR](20) NULL,
);
--創建設備表
CREATE TABLE [dbo].[Device]
([DeviceId]   [INT] IDENTITY(1, 1) NOT NULL, 
 [id]         [INT] NULL, 
 [DeviceName] [NVARCHAR](20) NULL
);

1.首先NuGet 安裝EntityFramework,至於什麼版本看一下介紹,選擇對應框架的版本,我的是4.8安裝的EF版本是6.2

2.在配置文件中設置連接字元串

<connectionStrings>
	<add name="efConstr" connectionString="Data Source=192.168.0.106;Initial Catalog=EfDemo;User ID=sa;Password=sa@123456" providerName="System.Data.SqlClient" />
</connectionStrings>

3.創建員工實體類和設備實體類,並添加相關屬性,添加2個方法添加和刪除Device信息

[Table("Employee")] 表映射
[Key] 主鍵
[Required] 必填
[Column("Name")] 列名映射
[StringLength(1000)] 設置長度,如果是dbfirst記得與資料庫長度匹配

[Table("Employee")]
public class EmployeeEntity
{
    public EmployeeEntity()
    {
        DeviceEntities = new HashSet<DeviceEntity>();
    }
    
    [DatabaseGenerated(DatabaseGeneratedOption.Identity)]
    [Key]
    [Column(Order = 1)]
    [Required]
    public int id { get; set; }

    //將資料庫欄位Name映射別名為"mingzi "
    [Column("Name")]
    public string mingzi { get; set; }

    public string Code { get; set; }

    public virtual ICollection<DeviceEntity> DeviceEntities { get; set; }

    public void AddDevice(DeviceEntity entity)
    { 
        this.DeviceEntities.Add(entity);
    }

    public void RemoveDevice(DeviceEntity entity)
    {
        this.DeviceEntities.Remove(entity);
    }
}

[Table("Device")]
public class DeviceEntity
{
    [DatabaseGenerated(DatabaseGeneratedOption.Identity)]
    [Key]
    public int DeviceId { get; set; }
    public int id { get; set; }
    public string DeviceName { get; set; }
}

4.創建UserContext上下文,繼承自DbContext

public class UserContext: DbContext
{
    public UserContext()
          : base("name=efConstr")
    {
    }
    public virtual DbSet<EmployeeEntity> EmployeeEntities { get; set; }
}

2.EF查詢

查詢員工和設備數據,為了查詢方便已經在實體模型上設置了,所以不需要連表查詢,直接根據員工查出關聯表的數據,我們想在程式中輸出ef執行的sql日誌可以使用

userContext.Database.Log = sql => { Console.WriteLine(sql); };

1.使用Find即時 查詢id為1的數據,此時執行代碼是直接實時查詢資料庫。

 using (UserContext userContext = new UserContext())
 {
     var result = userContext.EmployeeEntities.Find(1);
 }

2.使用Where延時查詢 查詢id大於0的數據,在執行迴圈之前此時還未提交到資料庫,得到的是一個IQueryable可以使用ToList()或者對IQueryable進行迴圈就會直接提交到資料庫
但是不建議直接使用ToList

public static List<EmployeeDto> SerachEmployeeInfo()
{
    List<EmployeeDto> employees = new List<EmployeeDto>();
    using (UserContext userContext = new UserContext())
    {
        //是一個IQueryable,此時並不會提交到資料庫
        var result =  userContext.EmployeeEntities.Where(x => x.id > 0);
        //雖然有多個但是只是拼接表達式,ToList時提交
        result = result.Where(x => x.id == 2).ToList();

        foreach (var item in result)
        {
            EmployeeDto employeeDto = new EmployeeDto();
            employeeDto.Id = item.id;
            employeeDto.Name = item.mingzi;
            employeeDto.Code = item.Code;
            employeeDto.DeviceEntities = item.DeviceEntities.ToList();
            employees.Add(employeeDto);
        }
    }
    return employees;
}

3.使用連接查詢,預設為內連,如果需要左連接就需要將2個合併插入一個新的表,然後使用DefaultIfEmpty() 定義可為空

 public static List<EmployeeDto> SerachEmployeeInfoByJoin()
 {
     List<EmployeeDto> employees = new List<EmployeeDto>();
     using (UserContext userContext = new UserContext())
     {
         List<int> ins = new List<int> { 2 };
         //使用內連接查詢
         var result = from u in userContext.EmployeeEntities
                      join c in userContext.DeviceEntities on u.id equals c.id
                      select new
                      {
                          Id = u.id,
                          Name = u.mingzi,
                          DeviceName = c.DeviceName
                      };
        //使用左連接查詢
        var result = from u in userContext.EmployeeEntities
                      join c in userContext.DeviceEntities on u.id equals c.id
                      into leftTable
                      from lt in leftTable.DefaultIfEmpty() 
                      select new
                      {
                          Id = u.id,
                          Name = u.mingzi,
                          DeviceName = c.DeviceName
                      };
         //提交到資料庫,並且遍歷投影出的新的數據
         foreach (var item in result)
         {}
     }
     return employees;
 }

4.使用EF執行sql語句,適合比較複雜的sql語句,主要依靠ExecuteSqlCommandSqlQuery,調用存儲過程也是同樣的道理

public static int SerachEmployeeInfoBySql ()
{
    using (UserContext userContext = new UserContext())
    {
        DbContextTransaction trans = null;
        try
        {
             trans = userContext.Database.BeginTransaction();
             string sqlExecute = "Update Employee set Name ='EF測試'  WHERE Id=@Id";
             //查詢語句
             //string sqlQuery = "Select*from Employee  WHERE Id=@Id"
             SqlParameter parameter = new SqlParameter("@Id", 1);
             //執行ExecuteSqlCommand增刪改
             int executeResult = userContext.Database.ExecuteSqlCommand(sqlExecute, parameter);
             //var result  = userContext.Database.SqlQuery<EmployeeEntity>(sqlQuery, parameter).ToList<EmployeeEntity>();
             trans.Commit();
             return executeResult;
        }
        catch (Exception ex)
        {
            if (trans != null)
                trans.Rollback();
            throw ex;
        }
        finally
        {
            trans.Dispose();
        }

    }
}

3.EF增刪改使用

1.新增一筆數據,這裡使用的是非同步方法,其實同步方法一樣的道理,由於EmployeeDevice是主從關係,所以執行下麵的代碼同時會向Device表也插入一條

  • 1.如果在一個上下文中再次將mingzi 改為"同一上下文中再次賦值",那麼就會修改前面插入的數據,因為在一個上下文中會進行數據跟蹤
static async Task AddEmployee()
{
     using (UserContext userContext = new UserContext())
     {
         EmployeeEntity entity = new EmployeeEntity();
         entity.mingzi = "新增數據1";
         entity.Code = "A123";
         //導航屬性插入值
         entity.AddDevice(new DeviceEntity { DeviceName = "ipad" });
         userContext.EmployeeEntities.Add(entity);
         //執行SaveChangesAsync()數據才會提交
         await userContext.SaveChangesAsync();

         //同上下文中再次賦值,就會修改mingzi的值
         entity.mingzi = "同一上下文中再次賦值";
         await userContext.SaveChangesAsync();
     }
     return result;
}

2.使用EF修改數據,如果直接修改是無效的

  • 2.1.在同一上下文中修改,通常在工作中的做法是需要先把對象查詢出來,然後修改對應的值。

同一上下文修改

static async Task UpdateEmployee()
{
    using (UserContext userContext = new UserContext())
    {
        //先查詢在修改
        var content = await userContext.EmployeeEntities.FindAsync(7);
        content.mingzi = "被修改的值";
        await userContext.SaveChangesAsync();
    }
}

  • 2.2.在不同上下文中修改值,我們使用Attach的目的就是把一個沒有被dbContext 跟蹤的對象附加到新的上下文中,使其被跟蹤,必須先附加再賦值才能生效,如果先賦值那就需要在上下文中將對象的狀態設置為EntityState.Modified
    不同上下文修改
static async Task AddEmployee()
{
    EmployeeEntity entity = null;
    using (UserContext userContext = new UserContext())
    {
        entity = new EmployeeEntity();
        entity.mingzi = "新增數據1";
        entity.Code = "A123";
        entity.AddDevice(new DeviceEntity { DeviceName = "ipad1" });
        userContext.EmployeeEntities.Add(entity);
        await userContext.SaveChangesAsync();
    }
    using (UserContext userContext = new UserContext())
    {
        //先附加再修改
        userContext.EmployeeEntities.Attach(entity);
        entity.mingzi = "不同上下文先附加再修改值";
        await userContext.SaveChangesAsync();

        //先修改再設置狀態為Modified
        entity.mingzi = "不同上下文先修改值再設置狀態";
        userContext.Entry<EmployeeEntity>(entity).State = EntityState.Modified;
        await userContext.SaveChangesAsync();
    }
}

  • 2.3.按照上面的方式,EF在執行時生成的sql會將所有的欄位都更新一遍,我們可以設置只對某個屬性欄位更新,在執行sql時只會更新那一個欄位,前提需要對對象進行Attach,然後再設置欄位狀態。
using (UserContext userContext = new UserContext())
{
    entity.mingzi = "不同上下文修改值";
    userContext.EmployeeEntities.Attach(entity);
    //通知context mingzi屬性被修改
    userContext.Entry<EmployeeEntity>(entity).Property<string>("mingzi").IsModified = true;
    await userContext.SaveChangesAsync();
}

3.刪除數據的方式和修改類似,通用的是查詢出來然後是使用Remove,如果不同上下文,我們需要Attach或者將狀態更改為EntityState.Deleted

static async Task removeEmployee(int id)
{
    using (UserContext userContext = new UserContext())
    {
        var emp = userContext.EmployeeEntities.Where(x => x.id == id).FirstOrDefault();
        userContext.EmployeeEntities.Remove(emp);
        await userContext.SaveChangesAsync();
    }
}

  • 3.1.不同上下文刪除使用附加或者設置狀態
static async Task removeEmployee(int id)
{
    EmployeeEntity entity = null;
    using (UserContext userContext = new UserContext())
    {
        entity = userContext.EmployeeEntities.Where(x => x.id == id).FirstOrDefault();
    }
    using (UserContext userContext = new UserContext())
    {
        //設置實體狀態為刪除
        // userContext.Entry(entity).State = EntityState.Deleted;
        //附加到當前上下文
        userContext.EmployeeEntities.Attach(entity);
        //執行刪除
        userContext.EmployeeEntities.Remove(entity);
        await userContext.SaveChangesAsync();
    }
}
4.Context上下文

1.下麵在一個上下文裡面執行多個操作,一次SaveChanges()是保存全部的變化,假如中間有一個失敗,那所有的都將失敗,可以理解到SaveChanges是事務的結束,所以不能全局使用

static void UpdateEmployee()
{
    using (UserContext userContext = new UserContext())
    {
        try
        {
            var content = userContext.EmployeeEntities.Find(22);
            content.mingzi = "一個上下文修改第一個值";
            var content1 = userContext.EmployeeEntities.Find(23);
            content1.mingzi = "一個上下文修改第二個值撒啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊撒啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊";
            userContext.SaveChanges();
        }
        catch (Exception ex) { throw ex; }
      
    }
}

2.不同的Context不能連接查詢,除非load到記憶體之後再去操作,如果是多線程或者多個請求,最好多個Context

5.查詢本地緩存

1.在EF中查詢是有緩存的,例如下麵在使用Where時每一次都會去資料庫查詢,但是使用Find時,他會現在本地記憶體中查找一次,如果前面查詢有結果,那就直接使用,那麼就有可能產生臟讀,但是對性能提升有幫助,使用時根據自身需求而定。

using (UserContext userContext = new UserContext())
{
    var result = userContext.EmployeeEntities.Where(x => x.id < 3).ToList();
    var result1 = userContext.EmployeeEntities.Where(x => x.id ==2).ToList();
    var result2 = userContext.EmployeeEntities.Find(1);
    var result3 = userContext.EmployeeEntities.Where(x => x.id == 1).ToList();
}

2.如果不希望查詢出的結果在記憶體中緩存,可以使用AsNoTracking不在記憶體中拷貝副本,直接返回,就算後續使用Find也不會讀取記憶體

using (UserContext userContext = new UserContext())
{
    var result = userContext.EmployeeEntities.Where(x => x.id < 3).AsNoTracking().ToList();
}
``

###### 6.導航屬性以及延遲載入

1.主從查詢丟棄子表查詢,在上下文中使用`LazyLoadingEnabled =fasle`

```cs
userContext.Configuration.LazyLoadingEnabled = false;

2.使用Include,不延遲載入將數據一次性載入出來

using (UserContext userContext = new UserContext())
{
    var result = userContext.EmployeeEntities.Include("DeviceEntities").Where(x=>x.id>0);
    foreach (var item in result)
    {
        EmployeeDto employeeDto = new EmployeeDto();
        employeeDto.Id = item.id;
        employeeDto.Name = item.mingzi;
        employeeDto.Code = item.Code;
        employeeDto.DeviceEntities = item.DeviceEntities.ToList(); 
        employees.Add(employeeDto);
    }
}

3.主從表的級聯刪除,前提是需要修改資料庫外鍵的刪除規則為級聯

 using (UserContext userContext = new UserContext())
 {
     entity = userContext.EmployeeEntities.Where(x => x.id == id).FirstOrDefault();
     userContext.EmployeeEntities.Remove(entity);
     userContext.SaveChanges();
 }


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

-Advertisement-
Play Games
更多相關文章
  • 怎麼能不懷念呢?到現在依稀記得高考時的自己,輾轉失眠、挑燈夜戰、奮筆勤書。那時候拼命的自己真的很好,好想懷念高考,加爬一下自己19年的高考數據吧。 對於像作者一樣已經工作的“上班族”來說,6月7號到9號三天無疑是興奮到飛起的,終於迎來了令人愉悅的端午假期: 然而有那麼一群人,將在端午節日之際迎來人生 ...
  • 前言 以往的人臉識別主要是包括人臉圖像採集、人臉識別預處理、身份確認、身份查找等技術和系統。現在人臉識別已經慢慢延伸到了ADAS中的駕駛員檢測、行人跟蹤、甚至到了動態物體的跟蹤。 由此可以看出,人臉識別系統已經由簡單的圖像處理髮展到了視頻實時處理。而且演算法已經由以前的Adaboots、PCA等傳統的 ...
  • 1 基礎知識 單點登錄機制(SSO)允許用戶登錄應用程式一次,並訪問所有相關的系統,而不需要單獨登錄它們。 由於 SSO,用戶只需登錄一次即可使用服務,並自動登錄到所有相關應用程式。SSO 消除了單獨登錄它們的需要。此外,用戶不需要存儲或記住多個憑據。 單點登錄僅與身份驗證過程相關。它的任務是驗證用 ...
  • 昨天,在發佈了《Spring官宣承認網傳大漏洞,並提供解決方案》之後。群里就有幾個小伙伴問了這樣的問題:**我們的Spring版本比較老,該怎麼辦?**這是一個好問題,所以DD今天單獨拿出來說說。 這次的RCE漏洞宣佈之後,官方給出的主要解決方案是升級版本,但只有Spring 5.2、5.3和Spr ...
  • 一、概要 1、來源 積木報表版本號:1.4.2 2、異常現象 導出Excel失敗 3、異常棧: com.googlecode.aviator.exception.CompileExpressionErrorException: Blank expression at com.googlecode.a ...
  • 前言 怎麼樣使用Python提高自己的工作效率,今天就給大家分享這個吧。我們經常用pandas讀入讀寫excel文件,經常會遇到一個excel文件里存在多個sheet文件,這個時候,就需要一次性讀取多個sheet的數據並做相應的數據分析或數據處理,最後再寫入新的excel文件(同樣是多個sheet) ...
  • 博客推行版本更新,成果積累制度,已經寫過的博客還會再次更新,不斷地琢磨,高質量高數量都是要追求的,工匠精神是學習必不可少的精神。因此,大家有何建議歡迎在評論區踴躍發言,你們的支持是我最大的動力,你們敢投,我就敢肝 ...
  • 05-Arrays, Slices, and Maps、 In memory、 Array、 Slice、 fence post error、 Compare Array and Slice 、 Map、 Built in functi... ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...