【ASP.NET Core學習】Entity Framework Core

来源:https://www.cnblogs.com/WilsonPan/archive/2019/11/11/11792722.html
-Advertisement-
Play Games

【ASP.NET Core學習】在ASP.NET Core 種使用Entity Framework Core介紹,包括如何添加Entity Framwork Core,創建模型和遷移到資料庫,查詢數據,保存數據,使用事務,處理併發衝突 ...


 這裡介紹在ASP.NET Core中使用EF Core,這裡資料庫選的是Sql Server

  1. 如何使用Sql Server
  2. 添加模型 && 資料庫遷移
  3. 查詢數據
  4. 保存數據

如何使用Sql Server

 1. 安裝dotnet-ef(已經安裝忽略)
dotnet tool install --global dotnet-ef

2. 添加包Microsoft.EntityFrameworkCore.Design

dotnet add package Microsoft.EntityFrameworkCore.Design

3. 添加包Microsoft.EntityFrameworkCore.SqlServer

dotnet add package Microsoft.EntityFrameworkCore.SqlServer

4. 添加DbContext

public class EFCoreDbContext : DbContext
                        {
                            public EFCoreDbContext(DbContextOptions<EFCoreDbContext> options)
                                : base(options)
                            {
                        
                            }
                        }
View Code

5.在ConfigureServices註入DbContext

public void ConfigureServices(IServiceCollection services)
                        {
                            services.AddRazorPages();
                        
                            services.AddDbContext<Data.EFCoreDbContext>(options =>
                                options.UseSqlServer(Configuration.GetConnectionString("DefaultConnection")));
                        }
View Code

經過上面5步,我們就可以在項目中使用資料庫,在需要的地方註入DbContext即可

 

添加模型

 我們就以學校 -> 學生這樣的模型(一對多)為例,欄位也儘量簡潔,這裡不是展示設計,以展示操作EF Core為主,所以類定義未必是最合適的。 學校類
[Table("School")]
                    public class School
                    {
                        [Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]
                        public int Id { get; set; }
                    
                        [Display(Name = "學校名稱")]
                        [Required(ErrorMessage = "學校名稱不能為空")]
                        [StringLength(100, ErrorMessage = "學校名稱最大長度為100")]
                        public string Name { get; set; }
                    
                        [Display(Name = "學校地址")]
                        [Required(ErrorMessage = "學校地址不能為空")]
                        [StringLength(200, ErrorMessage = "學校地址最大長度為200")]
                        public string Address { get; set; }
                    
                        public List<Student> Students { get; set; }
                    
                        [Display(Name = "創建時間")]
                        [DataType(DataType.DateTime), DisplayFormat(DataFormatString = "{0:yyyy-MM-dd HH:mm:ss}")]
                        public DateTime CreateTime { get; set; }
                    
                        [Display(Name = "最後更新時間")]
                        [DataType(DataType.DateTime), DisplayFormat(DataFormatString = "{0:yyyy-MM-dd HH:mm:ss}")]
                        public DateTime? LastUpdateTime { get; set; }
                    
                    }
View Code

 學生類

public class Student
                    {
                        [Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]
                        public int Id { get; set; }
                    
                        [Display(Name = "學生姓名")]
                        [Required(ErrorMessage = "學生姓名不能為空")]
                        [StringLength(50, ErrorMessage = "學生姓名最大長度為50")]
                        public string Name { get; set; }
                    
                        [Display(Name = "年齡")]
                        [Required(ErrorMessage = "年齡不能為空")]
                        [Range(minimum: 10, maximum: 100, ErrorMessage = "學生年齡必須在(10 ~ 100)之間")]
                        public int Age { get; set; }
                    
                        public School School { get; set; }
                    
                        [Display(Name = "創建時間")]
                        [DataType(DataType.DateTime), DisplayFormat(DataFormatString = "{0:yyyy-MM-dd HH:mm:ss}")]
                        public DateTime CreateTime { get; set; }
                    
                        [Display(Name = "最後更新時間")]
                        [DataType(DataType.DateTime), DisplayFormat(DataFormatString = "{0:yyyy-MM-dd HH:mm:ss}")]
                        public DateTime? LastUpdateTime { get; set; }
                    }
View Code

 配置預設值

protected override void OnModelCreating(ModelBuilder modelBuilder)
                    {
                        modelBuilder.Entity<Models.School>()
                                    .Property(p => p.CreateTime)
                                    .HasDefaultValueSql("getdate()");
                    
                        modelBuilder.Entity<Models.Student>()
                                    .Property(p => p.CreateTime)
                                    .HasDefaultValueSql("getdate()");
                    }
View Code

 

模型定義好之後,我們需要把模型添加到DbContext

public DbSet<Models.School> Schools{ get; set; }
                    public DbSet<Models.Student> Students { get; set; }
然後需要更新模型到資料庫,執行下麵兩條命令 1. 新增一個遷移
dotnet ef migrations add DatabaseInit

 2. 更新到資料庫

dotnet ef migrations add DatabaseInit

查看資料庫,我們可以看到下麵關係圖

 

在Student表裡面多了一個SchoolId,這個我們是沒有定義,是EF Core生成的陰影屬性,當然我們也可以顯示定義這個欄位

實體類定義我們用到數據註釋和Fluent API約束實體類生成,下麵列取經常用到的

註釋用途
 Key  主鍵
 Required  必須
 MaxLength   最大長度
NotMapped 不映射到資料庫
ConcurrencyCheck 併發檢查
Timestamp 時間戳欄位
 

 

查詢數據

一、聯接查詢
var query = from a in _context.School
                        join b in _context.Student on a.Id equals b.School.Id
                        select new
                        {
                            SchoolName = a.Name,
                            StudentName = b.Name
                        };
View Code

 對應生成的Sql

SELECT [s].[Name] AS [SchoolName], [t].[Name] AS [StudentName]
                  FROM [School] AS [s]
                  INNER JOIN (
                      SELECT [s0].[Id], [s0].[Age], [s0].[CreateTime], [s0].[LastUpdateTime], [s0].[Name], [s0].[SchoolId], [s1].[Id] AS [Id0], [s1].[Address], [s1].[CreateTime] AS [CreateTime0], [s1].[LastUpdateTime] AS [LastUpdateTime0], [s1].[Name] AS [Name0]
                      FROM [Student] AS [s0]
                      LEFT JOIN [School] AS [s1] ON [s0].[SchoolId] = [s1].[Id]
                  ) AS [t] ON [s].[Id] = [t].[Id0]

和我們預期有點不一致,預期是兩個表的全連接,為什麼出現這個,原因是Student裡面的導航屬性School,Linq遇到導航屬性是通過連表得到,為了驗證這個,我們不使用陰影屬性,顯示加上SchoolId試試

var query = from a in _context.School
                        join b in _context.Student on a.Id equals b.SchoolId
                        select new
                        {
                            SchoolName = a.Name,
                            StudentName = b.Name
                        };
View Code

對應生成的Sql

SELECT [s].[Name] AS [SchoolName], [s0].[Name] AS [StudentName]
            FROM [School] AS [s]
            INNER JOIN [Student] AS [s0] ON [s].[Id] = [s0].[SchoolId]

這次生成的Sql就很簡潔,跟預期一樣,所以如果使用聯接查詢,最好是避免使用陰影屬性

兩個Sql的執行計劃

 二、GroupBy查詢

var query = from a in _context.School
                    join b in _context.Student on a.Id equals b.SchoolId
                    group a by a.Name into t
                    where t.Count() > 0
                    orderby t.Key
                    select new
                    {
                        t.Key,
                        Count = t.Count(),
                    };
View Code

對應生成的Sql

SELECT [s].[Name] AS [Key], COUNT(*) AS [Count]
        FROM [School] AS [s]
        INNER JOIN [Student] AS [s0] ON [s].[Id] = [s0].[SchoolId]
        GROUP BY [s].[Name]
        HAVING COUNT(*) > 0
        ORDER BY [s].[Name]

EF Core 支持的聚合運算符如下所示

  • 平均值
  • 計數
  • LongCount
  • 最大值
  • 最小值
  • Sum

 三、左連接

var query = from a in _context.School
                    join b in _context.Student on a.Id equals b.SchoolId into t1
                    from t in t1.DefaultIfEmpty()
                    select new
                    {
                        SchoolName = a.Name,
                        StudentName = t.Name
                    };
        
        var list = query.AsNoTracking().ToList();
View Code

對應生成的Sql

SELECT [s].[Name] AS [SchoolName], [s0].[Name] AS [StudentName]
        FROM [School] AS [s]
        LEFT JOIN [Student] AS [s0] ON [s].[Id] = [s0].[SchoolId]

四、小結

全聯接時避免使用導航屬性連表

預設情況是跟蹤查詢,這表示可以更改這些實體實例,然後通過 SaveChanges() 持久化這些更改,

如果只需要讀取,不需要修改可以指定非跟蹤查詢AsNoTracking

非跟蹤查詢可以在每個查詢後面指定,還可以在上下文實例級別更改預設跟蹤行為

context.ChangeTracker.QueryTrackingBehavior = QueryTrackingBehavior.NoTracking;

 

保存數據

一、關聯數據
_context.School.Add(new Models.School
            {
                Name = "暨南大學",
                Address = "廣州市黃埔大道西601號",
                Students = new System.Collections.Generic.List<Models.Student>()
                {
                    new Models.Student
                    {
                        Name= "黃偉",
                        Age = 21,
                    },
                },
            });
    
    _context.SaveChanges();
View Code

 同時在School,Student表保存數據,自動維護Student表的SchoolId欄位數據

 

二、級聯刪除

var school = _context.School.Include(m => m.Students).FirstOrDefault(m => m.Name == "濟南大學");
    
    _context.School.Remove(school);
    
    _context.SaveChanges();
View Code 對應生成的Sql
--1. 讀取濟南大學和他所有學生
    SELECT [t].[Id], [t].[Address], [t].[CreateTime], [t].[LastUpdateTime], [t].[Name], [s0].[Id], [s0].[Age], [s0].[CreateTime], [s0].[LastUpdateTime], [s0].[Name], [s0].[SchoolId]
    FROM (
        SELECT TOP(1) [s].[Id], [s].[Address], [s].[CreateTime], [s].[LastUpdateTime], [s].[Name]
        FROM [School] AS [s]
        WHERE [s].[Name] = N'濟南大學'
    ) AS [t]
    LEFT JOIN [Student] AS [s0] ON [t].[Id] = [s0].[SchoolId]
    ORDER BY [t].[Id], [s0].[Id]
    
    --2. 迴圈每個學生刪除
    SET NOCOUNT ON;
    DELETE FROM [Student]
    WHERE [Id] = @p0;
    SELECT @@ROWCOUNT;
    SET NOCOUNT ON;
    DELETE FROM [Student]
    WHERE [Id] = @p0;
    SELECT @@ROWCOUNT;
    SET NOCOUNT ON;
    DELETE FROM [Student]
    WHERE [Id] = @p0;
    SELECT @@ROWCOUNT;
    
    --3. 刪除學校
    SET NOCOUNT ON;
    DELETE FROM [School]
    WHERE [Id] = @p1;
    SELECT @@ROWCOUNT;
View Code

級聯刪除要用Include把子項也包含到實體

 

三、使用事務

預設情況下,如果資料庫提供程式支持事務,則會在事務中應用對 SaveChanges() 的單一調用中的所有更改。 如果其中有任何更改失敗,則會回滾事務且所有更改都不會應用到資料庫。 這意味著,SaveChanges() 可保證完全成功,或在出現錯誤時不修改資料庫。

對於大多數應用程式,此預設行為已足夠。 如果應用程式要求被視為有必要,則應該僅手動控制事務中間調用多次SaveChanges()也不會直接保存到資料庫,最後transaction.Commit()

using (var transaction = _context.Database.BeginTransaction())
    {
        var school = _context.School.Add(new Models.School
        {
            Name = "濟南大學",
            Address = "山東省濟南市南辛莊西路336號",
        });
        _context.SaveChanges();
    
        System.Threading.Thread.Sleep(2000);  //for testing
        _context.Student.Add(new Models.Student
        {
            Name = "張三",
            Age = 29,
            School = school.Entity
        });
        _context.SaveChanges();
    
        transaction.Commit();
    }
View Code

 下麵是Sql Server Profiler

 

註意兩次RPC:Completed時間,每次調用SaveChanges提交到資料庫執行,外麵包一層事務,所以事務裡面要儘可能的控制操作最少,時間最少

 

四、併發衝突

EF Core實現的是樂觀併發,有關樂觀併發和悲觀併發這裡就不展開。

EF處理併發分兩種情況,單個屬性併發檢查和時間戳(又叫行版本),單個屬性只保證單個欄位併發修改,時間戳是保證整條數據的併發修改

我們在Student的Age加上[ConcurrencyCheck],在School加上行版本

[ConcurrencyCheck]
public int Age { get; set; }
[Timestamp]
public byte[] RowVersion { get; set; }

1. 模擬Age併發衝突

var student = _context.Student.Single(m => m.Id == 1);
student.Age = 32;

#region 模擬另外一個用戶修改了Age

var task = Task.Run(() =>
{
    var options = HttpContext.RequestServices.GetService<DbContextOptions<Data.EFCoreDbContext>>();
    using (var context = new Data.EFCoreDbContext(options))
    {
        var student = context.Student.Single(m => m.Id == 1);
        student.Age = 23;
        context.SaveChanges();
    }
});
task.Wait();

#endregion

try
{
    _context.SaveChanges();
}
catch (DbUpdateConcurrencyException ex)
{
    _logger.LogError(ex, "database update error");
}
View Code

2. 資料庫數據

 可以看到是Task裡面的更新成功了

3. 異常信息

Database operation expected to affect 1 row(s) but actually affected 0 row(s). Data may have been modified or deleted since entities were loaded

異常信息描述很明確,就是資料庫操作期望1行被影響,實際是0行,數據可能被修改或刪除自從實體載入後

4. SQL

exec sp_executesql N'SET NOCOUNT ON;
UPDATE [Student] SET [Age] = @p0
WHERE [Id] = @p1 AND [Age] = @p2;
SELECT @@ROWCOUNT;

',N'@p1 int,@p0 int,@p2 int',@p1=1,@p0=23,@p2=25

exec sp_executesql N'SET NOCOUNT ON;
UPDATE [Student] SET [Age] = @p0
WHERE [Id] = @p1 AND [Age] = @p2;
SELECT @@ROWCOUNT;

',N'@p1 int,@p0 int,@p2 int',@p1=1,@p0=32,@p2=25
View Code

加上併發檢查的欄位會在where條件後面加上原始值,Timestamp也是一樣道理,只是Timestamp是每次(插入/更新)資料庫會更新這個欄位,數字遞增的形式。

5. 解決併發衝突

要解決上面衝突,先要介紹EF Core裡面三組數值

原始值:實體從資料庫載入時的值   (例子:Age = 25)

當前值:實體當前的值        (例子:Age = 32)

資料庫值:當前資料庫中的值     (例子:Age = 23)

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

-Advertisement-
Play Games
更多相關文章
  • 背景 之前做的海量數據數據展示,在預處理速度和線上渲染上還有有所欠缺,本文中進行一些優化工作,使得九分鐘處理完一千多萬面數據的3 12級矢量切片,線上瀏覽數據請求時間控制在10s左右。 準備 軟體環境:PostGIS(3.0.0rc2 r17909)和 PostgreSQL( 12.0, compi ...
  • Eclipse部署多模塊項目到tomcat,啟動時找不到jar的解決方法。 ...
  • 在使用redis時,一般會設置一個過期時間,當然也有不設置過期時間的,也就是永久不過期。當設置了過期時間,redis是如何判斷是否過期,以及根據什麼策略來進行刪除的。 設置過期時間 expire key time(以秒為單位) 這是最常用的方式setex(String key, int second ...
  • scp是secure copy的簡寫,用於在Linux下進行遠程拷貝文件的命令,和它類似的命令有cp,不過cp只是在本機進行拷貝不能跨伺服器,而且scp傳輸是加密的。可能會稍微影響一下速度。當你伺服器硬碟變為只讀 read only system時,用scp可以幫你把文件移出來。另外,scp還非常不 ...
  • 你可能想創建一個在應用的任何地方都可以訪問的函數,這個教程將幫你實現 👏 很多教程都會說,你在 composer.json 這個文件中通過添加一個自動載入的文件,就可以實現這個需求。但我認為這不是一個好的方式,當你在 helpers.php 文件中添加了更多的函數時,可讀性將變得很差。 下麵我將介 ...
  • 在說正題之前先解釋一下交換機模式是個籠統的稱呼,它不是一個單獨的模式(包括了訂閱模式,路由模式和主題模式),交換機模式是一個比較常用的模式,主要是為了實現數據的同步。 首先,說一下訂閱模式,就和字面上的意思差不多主要就是一個生產者,多個消費者,同一個消息被多個消費者獲取,先看一下官網的圖示 整體執行 ...
  • 當創建隊列jobs、監聽器或訂閱伺服器以推送到隊列中時,您可能會開始認為,一旦分派,隊列工作器決定如何處理您的邏輯就完全由您自己決定了。 嗯……並不是說你不能從作業內部與隊列工作器交互,但是通常情況下,哪怕你做了,也是沒必要的。 這個神奇的騷操作的出現是因為“InteractsWithQueue”這 ...
  • 環境:MacOS 10.13 MAMAP Prophp 7.0.33 + xdebugVisual Studio Code前言我所理解的 POP Chain:利用魔術方法並巧妙構造特殊屬性調用一系列函數或類方法以執行某種敏感操作的調用堆棧反序列化常用魔法函數前言我所理解的 POP Chain:利用魔 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...