asp.net core系列 29 EF模型配置(查詢類型,關係資料庫建模)

来源:https://www.cnblogs.com/MrHSR/archive/2019/02/21/10411284.html
-Advertisement-
Play Games

一.查詢類型 此功能是EF Core 2.1中的新功能。 EF Core除了實體類型之外,EF Core模型還可以包含查詢類型,這些查詢類型是針對“未映射到實體類型”的數據獲取。比如視圖,或只讀數據表。 1.1 下麵介紹下,查詢類型與實體類型共同與不同點: (1) 可以在OnModelCreatin ...


一.查詢類型

  此功能是EF Core 2.1中的新功能。 EF Core除了實體類型之外,EF Core模型還可以包含查詢類型,這些查詢類型是針對“未映射到實體類型”的數據獲取。比如視圖,或只讀數據表。

  

  1.1 下麵介紹下,查詢類型與實體類型共同與不同

    (1) 可以在OnModelCreating中或通過派生DbContext上的“set”屬性添加到EF Core模型中。

    (2) 支持許多相同的映射功能,在關係資料庫上,如繼承映射和導航屬性。也可以配置目標資料庫對象和列通過 fluent API 方法或數據註釋。

    但是,它們不同於實體中的類型:

    (1) 不需要定義的鍵。

         (2) 不會跟蹤DbContext上的更改,因此不會在資料庫中插入、更新或刪除。

    (3) 永遠不會由約定發現。

    (4) 僅支持一部分導航映射功能

                     它們可能永遠不會作為關係的主體端(沒有主體實體和依賴實體的關係)。

                     它們僅可包含指向的實體的引用導航屬性。

                     實體不能包含查詢類型的導航屬性。

    (5) 在ModelBuilder上使用Query方法而不是Entity方法進行定址。

    (6) 是否通過類型DbQuery<T>而不是DbSet<T>的屬性映射到DbContext上。

    (7) 映射到使用的資料庫對象ToView方法,而非ToTable。

    (8) 可以映射到定義查詢。定義查詢是在模型中聲明的輔助查詢,充當查詢類型的數據。

 

  1.2 查詢類型使用方案(應用場景)

    (1) 作為返回類型的即席FromSql()查詢。

    (2) 映射到資料庫視圖。

    (3) 映射到不具有定義的主鍵的表。

    (4) 映射到模型中定義的查詢。

  

  1.3 映射到資料庫對象

    查詢類型映射到的資料庫對象,通過實現ToView fluent API此ToView方法中指定的資料庫對象,從 EF Core 的角度來看是視圖,這意味著它將被視為只讀查詢源和不能作為目標的更新、插入或刪除操作(也可以將被視為只讀表)。 相反,對於實體類型,EF Core 假定資料庫對象中指定ToTable方法可以視為表或視圖、 意味著它可以用作查詢源,還可以做更新、 刪除和插入操作。

      // 查詢類型的Blog ,  參數: view or table.
        modelBuilder.Query<Blog>().ToView("xx");
      // 實體類型的Blog    參數: view or table。註意如果是視圖,更新數據,只能是一個表的視圖。
        modelBuilder.Entity<Blog>().ToTable("xx");

 

  1.4 案例

    下麵的示例演示如何使用查詢類型來查詢資料庫視圖。完整代碼可查看官方示例(下麵有鏈接)。

     (1)首先定義一個簡單的Blog和Post實體類型。

public class Blog
{
    public int BlogId { get; set; }
    public string Name { get; set; }
    public string Url { get; set; }
    public ICollection<Post> Posts { get; set; }
}

public class Post
{
    public int PostId { get; set; }
    public string Title { get; set; }
    public string Content { get; set; }
    public int BlogId { get; set; }
}

    (2) 使用BloggingContext上下文生成一個簡單的資料庫視圖,這樣就可以查詢與每個Blog的帖子(Posts)數

          BloggingContext.Database.ExecuteSqlCommand(
                       @"CREATE VIEW View_BlogPostCounts AS 
                            SELECT b.Name, Count(p.PostId) as PostCount 
                            FROM Blogs b
                            JOIN Posts p on p.BlogId = b.BlogId
                            GROUP BY b.Name");

          

    (3) 接下來,我們定義一個查詢類來保存資料庫視圖的結果

public class BlogPostsCount
{
    public string BlogName { get; set; }
    public int PostCount { get; set; }
}

    (4) 我們配置中的查詢類型在OnModelCreating中使用modelBuilder.Query<T>API。 我們使用標準的 fluent 配置 Api 來配置查詢類型的映射:

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder
        .Query<BlogPostsCount>().ToView("View_BlogPostCounts")
        .Property(v => v.BlogName).HasColumnName("Name");
}

    (5) 最後,我們可以採用標準方式來查詢資料庫視圖:

  var postCounts = BloggingContext.BlogPostCounts.ToList();

    

二. 關係資料庫建模

   一般而言,本部分中的配置適用於關係資料庫。安裝關係資料庫提供程式時,下麵顯示的擴展方法將變為可用,原因在於共用Microsoft.EntityFrameworkCore.Relational包 

  

  2.1 表映射

    表映射是指(實體與數據表的映射)包括查詢的表數據,並將其保存到資料庫中。

    按照約定,每個實體都將映射到一個表,該表的名稱與DbSet<TEntity>在派生上下文中公開實體的屬性相同。例如在EF上下文中公開Blog類型,生成的表名與屬性名相同。

    //生成Blogs表名
    public DbSet<Blog> Blogs { get; set; }
    //除了約定還可以使用數據註釋,表blogs與實體Blog映射
    [Table("blogs")]
    public class Blog
    {
           public int BlogId { get; set; }
        public string Url { get; set; }
    }    
    
    //還可以使用Fluent API 配置,功能實現與上面一樣
     modelBuilder.Entity<Blog>().ToTable("blogs");

  

  2.2 列映射

    按照約定,實體中的每個屬性都會映射到表中,具有相同名稱的數據欄位。還可以使用數據註釋或Fluent API 配置:

public class Blog
{
   //數據註釋將表Blog的blog_id欄位映射到BlogId屬性。
    [Column("blog_id")]
    public int BlogId { get; set; }
    public string Url { get; set; }
}
    
//使用Fluent API配置,功能實現與上面一樣
 modelBuilder.Entity<Blog>().Property(b => b.BlogId).HasColumnName("blog_id");

 

  2.3 數據類型

     數據類型是指:資料庫的數據類型與 CLR 屬性類型映射。可以使用數據註釋來指定精確的數據類型的列(一般用在code first模式)。

//數據註釋的案例
public class Blog
{
    public int BlogId { get; set; }
    [Column(TypeName = "varchar(200)")]
    public string Url { get; set; }
    [Column(TypeName = "decimal(5, 2)")]
    public decimal Rating { get; set; }
}
//Fluent API配置,功能實現與上面一樣 modelBuilder.Entity<Blog>(eb => { eb.Property(b => b.Url).HasColumnType("varchar(200)"); eb.Property(b => b.Rating).HasColumnType("decimal(5, 2)"); });

 

  2.4 主鍵

    對於每個實體類型的鍵是引入了主鍵約束。按照約定,主鍵名稱是PK_<type name>,例如Posts 實體中有BlogId屬性,那表中主鍵約束名是PK_Posts。可以使用Fluent API配置資料庫中的主鍵約束的名稱。

  //將預設的PK_Posts改為了PrimaryKey_BlogId主鍵約束名
    modelBuilder.Entity<Blog>()
            .HasKey(b => b.BlogId)
            .HasName("PrimaryKey_BlogId");

 

   2.5 預設架構

    如果對象沒有顯式配置架構,則預設架構是創建對象的資料庫架構。按照約定,資料庫提供程式將選擇最合適的預設架構。例如,Microsoft SQL Server將使用該dbo模式。無法使用數據註釋設置預設架構,可以使用Fluent API指定預設架構。

    -- 修改sqlserver 表預設架構
    ALTER SCHEMA [blogging] TRANSFER dbo.Blogs
    -- 查詢表
    select * from [blogging].Blogs
      //設置ef core 對應資料庫表查詢的架構名
      modelBuilder.HasDefaultSchema("blogging");

 

  2.6 預設值

    如果插入新行,但未指定列值,可設置一個預設值。不可以使用約定或數據註釋提供預設值。可以使用Fluent API 配置預設值.

    //例如插入數據時,設置Created欄位取sqlserver的當前時間值。
      modelBuilder.Entity<Blog>()
            .Property(b => b.Created)
            .HasDefaultValueSql("getdate()");

 

  2.7 索引

    關係資料庫中的索引映射到相同的概念的 EF core 中的索引。按照約定,索引名為IX_<type name>_<property name>對於複合索引,下劃線分隔的屬性名稱列表。可以使用Fluent API配置索引的名稱。

      //設置url索引名稱為Index_Url
            modelBuilder.Entity<Blog>()
            .HasIndex(b => b.Url)
            .HasName("Index_Url");

      //還可以指定索引過濾器
            modelBuilder.Entity<Blog>()
            .HasIndex(b => b.Url)
            .HasFilter("[Url] IS NOT NULL");

 

 

  參考文獻:

    官方文檔:EF查詢類型

           查詢類型示例

           關係資料庫建模

 


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

-Advertisement-
Play Games
更多相關文章
  • json模塊 JSON (JavaScript Object Notation):是一個輕量級的數據交換格式模塊,受javascript對象文本語法啟發,但不屬於JavaScript的子集。 常用方法: dump(obj,fp):將對象以字元串的形式寫入文件中。 load(fp):將數據從文件中讀出 ...
  • H2engine的GitHub星星不知不覺已經破百了,也沒有特意推廣過,但是慢慢的關註的人越來越多。因為事情多,好久沒有寫東西了,前一段時間有了一些想法,把h2engine又更新了一下,感覺h2engine又向前邁了一大步。本文記錄一下最近的心得體會,以及做出的相應修改。 ...
  • 對於泛型的使用我想大家都非常熟悉,但是對於類型擦除,邊界拓展等細節問題,可能不是很清楚,所以本文會重點講解一下;另外對泛型的瞭解其實可以看出,一個語言特性的產生邏輯,這對我們平時的開發也是非常有幫助的; 一、為什麼會出現泛型 首先泛型並不是Java的語言特性,是直到 JDK1.5 才支持的特性(具體 ...
  • error介面1.error就是一個介面interface2.屬於errors包,該包有一個導出方法New,返回了errorString類型3.errorString類型實現了error介面4.之所以這樣是因為可以實現每個錯誤都是不同的實例 ...
  • 介面類型1.介面類型具體描述了一系列方法的集合,實現這些方法的具體類型是這個介面類型的實例2.一個類型如果擁有一個介面需要的所有方法,那麼這個類型就實現了這個介面 ...
  • ASP.NET core 的Filter是系統中經常用到的,本文詳細分享一下各種Filter定義、執行的內部機制以及執行順序。 一、 概述 ASP.NET Core MVC 中有好幾種常用的篩選器,例如Authorization filters 、Resource filters、Action fi ...
  • 在窗體的InitializeComponent();方法後面添加下麵的代碼。 MdiClient MC = new MdiClient(); MC.Name = "MdiClientForm"; MC.Dock = DockStyle.Fill; MC.BackgroundImageLayout = ...
  • 最近在修改自動化小工具,用多線程來解決後臺拷貝導致WinForm界面卡死的情況,但是遇到過錯:線程間操作無效: 從不是創建控制項“dataGridView”的線程訪問它。 這是因為在多線程程式中,新創建的線程不能訪問UI線程創建的視窗控制項,如果需要訪問視窗中的控制項,有2種解決方法: 1. 在Form_ ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...