asp.net core系列 27 EF模型配置(索引,備用鍵,繼承)

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

一.索引 索引是許多數據存儲中的常見概念。雖然它們在數據存儲中的實現可能會有所不同,但它們可用於更有效地基於列(或列集)進行查找。按照約定,用作外鍵每個屬性 (或組的屬性) 會自動創建索引。無法使用數據註釋創建索引。 1.1 非唯一索引 Fluent API 在單個屬性上指定索引。預設情況下,索引是 ...


一.索引

  索引是許多數據存儲中的常見概念。雖然它們在數據存儲中的實現可能會有所不同,但它們可用於更有效地基於列(或列集)進行查找。按照約定,用作外鍵每個屬性 (或組的屬性) 會自動創建索引。無法使用數據註釋創建索引。

  

  1.1 非唯一索引

    Fluent API 在單個屬性上指定索引。預設情況下,索引是非唯一的。如下代碼示例在Blogs表上創建Url列索引:

class MyContext : DbContext
{
    public DbSet<Blog> Blogs { get; set; }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Blog>()
            .HasIndex(b => b.Url);
    }
}

    

  1.2 唯一索引

    下麵代碼指定索引是唯一的,這是在索引上加了(Unique)唯一約束。

        modelBuilder.Entity<Blog>()
            .HasIndex(b => b.Url)
            .IsUnique();

 

  1.3 複合索引

 class MyContext : DbContext
{
    public DbSet<People> People { get; set; }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Person>()
            .HasIndex(p => new { p.FirstName, p.LastName });
    }
}

public class People
{
    public int PersonId { get; set; }
    public string FirstName { get; set; }
    public string LastName { get; set; }
    public string Address{get;set;}
}

    下麵使用EF基於數據模型(People)創建數據表。在Migration中生成了索引的代碼, 以及查看資料庫People表的索引(官方文檔中暫沒有看到提供索引包含列設置)如下所示:

         name: "IX_People_FirstName_LastName",
         table: "People",
         columns: new[] { "FirstName", "LastName" });

      

 

二.備用鍵

         除主鍵之外,備用鍵用作每個實體實例的備用唯一標識符(跟主鍵一樣具有唯一約束)。備用鍵可以用作關係的目標。當使用關係資料庫時,這映射到備用鍵列上的唯一索引/約束的概念以及引用列的一個或多個外鍵約束。系統通常會在需要時為你引入備用鍵,你無需手動配置它們。不能使用數據註釋配置備用鍵。

  

  2.1 約定

    按照約定,系統將在識別屬性(不是主鍵)時為你引入備用鍵,充當關係的目標。如下麵代碼所示:

class MyContext : DbContext
{
    public DbSet<Blog> Blogs { get; set; }
    public DbSet<Post> Posts { get; set; }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Post>()
            .HasOne(p => p.Blog)
            .WithMany(b => b.Posts)
             //Post中創建BlogUrl外建欄位
            .HasForeignKey(p => p.BlogUrl)
            //Blog中設置唯一約束備份鍵
            .HasPrincipalKey(b => b.Url);
    }
}

public class Blog
{
    public int BlogId { get; set; }
    public string Url { get; set; }

    public List<Post> Posts { get; set; }
}

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

    public string BlogUrl { get; set; }
    public Blog Blog { get; set; }
}

  上面主體實體Blog中Url屬性作為備用鍵,創建了AK_Blogs_Url唯一非聚集索引。在依賴實體Post中創建了BlogUrl外鍵欄位, 使用EF基於數據模型(Blog和Post實體)創建資料庫,如下圖所示。

  

  2.2 Fluent API

    可以使用Fluent API將單個屬性配置為備用鍵。

    class MyContext : DbContext
{
    public DbSet<Car> Cars { get; set; }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Car>()
            //配置備用鍵(唯一非聚集索引)
            .HasAlternateKey(c => c.LicensePlate);

            //  創建複合備用鍵
            //  modelBuilder.Entity<Car>()
            // .HasAlternateKey(c => new { c.State, c.LicensePlate });
    }
}

class Car
{
    public int CarId { get; set; }
    public string LicensePlate { get; set; }
    public string Make { get; set; }
    public string Model { get; set; }
    public string State { get; set; }
}

 

三.繼承

   EF 模型中的繼承用於控制如何在資料庫中表示實體類中的繼承, 按照約定,由資料庫提供程式決定如何在資料庫中表示繼承。有關如何使用關係資料庫提供程式處理它,請查看”繼承關係資料庫“。如果模型中明確包含兩個或多個繼承類型,EF將僅設置繼承。EF 不會掃描的基類或派生類型,可以在模型中包含類型,通過公開DbSet 繼承層次結構中每個類型。不能使用數據註釋來配置繼承。

  

  3.1 約定

     下麵示例中,有二個實體,通過公開Dbset類型,預設約定繼承,如下所示:

class MyContext : DbContext
{
    public DbSet<Blog> Blogs { get; set; }
    public DbSet<RssBlog> RssBlogs { get; set; }
}

public class Blog
{
    public int BlogId { get; set; }
    public string Url { get; set; }
}

public class RssBlog : Blog
{
    public string RssUrl { get; set; }
}

    使用EF基於數據模型(Blog和RssBlog實體)創建資料庫。生成後,兩個實體合併到一個Blogs表中,如下所示:

    

    

 

  3.2  Fluent API

    如果您不想公開DbSet對於層次結構中的一個或多個實體,您可以使用Fluent API確保它們包含在模型中。如果您不依賴約定,則可以使用明確指定基類型HasBaseType。 

class MyContext : DbContext
{
    public DbSet<Blog> Blogs { get; set; }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<RssBlog>().HasBaseType<Blog>();
    }
}

 

   3.3 discriminator隱藏屬性

     上面3.1示例中,創建了discriminator辨別者隱藏屬性,是基於base entity的層級。因為它是模型中的一個屬性,所以可以像配置其他屬性一樣配置它。例如,要設置預設情況下的最大長度。

modelBuilder.Entity<Blog>()
         .Property("Discriminator")
            .HasMaxLength(200);

    discriminator鑒別器也可以映射到實體中的實際CLR屬性

    class MyContext : DbContext
{
    public DbSet<Blog> Blogs { get; set; }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Blog>()
            .HasDiscriminator<string>("BlogType");
    }
}

public class Blog
{
    public int BlogId { get; set; }
    public string Url { get; set; }
    //discriminator
    public string BlogType { get; set; }
}

public class RssBlog : Blog
{
    public string RssUrl { get; set; }
}

 

  參考文獻:

    官方文檔:EF索引

           EF備用鍵

         EF繼承

         EF繼承(關係資料庫)

 


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

-Advertisement-
Play Games
更多相關文章
  • 寫在前面 WPF中常常有這樣的情況:需要在UI上顯示一些信息,比如顯示一張圖片的信息,信息結構是: 圖片名:Xxx 圖片尺寸:Xxx 而其中的 Xxx 通常通過數據綁定來獲得, Xxx 前面的內容是需要在xaml中寫死的,這個時候如何佈局比較方便呢? 可以使用StringFormat來簡單實這個需求 ...
  • 網路結構圖如下: 開題先放圖,一切全靠編哈哈。 進入正題! 如圖所示,我們需要一個公網伺服器,利用公網伺服器將內網的數據進行轉發,從而實現兩個內網伺服器的通訊。沒錯,這不是p2p,原本想做成p2p,因為有點難度,所以先弄一個tcp數據包轉發,後面再來打洞進行p2p連接。 開發環境 VS2017 + ...
  • emmmmm,最近在研究WFDB工具箱,C語言寫的,無奈本人C語言功底不夠,只想直接拿來用,於是打算通過ProcessStartInfo來調取編譯出來的exe程式獲取輸出。 一開始就打算偷懶,從園子里的前輩blog上偷來部分代碼,和著自己寫的代碼差不多就寫了以下調取程式的代碼: 然後使用時發現了問題 ...
  • ``` using Microsoft.Extensions.Configuration; using System; using System.Collections.Generic; using System.IO; ``` ``` public static class Configs { p... ...
  • 泛型的好處:類型安全性能高,代碼重用擴展好。 泛型的使用:如果我們需要使用多個泛型來實例化一個類型,那麼我們就需要使用說明性的名稱,比如TId,TFirstName之類的。 泛型的約束: where T : struct -類型T必須是值類型 where T : class -類型T必須是引用類型 ...
  • 一、背景 目前我們項目是採用的 Ocelot 作為 API 網關,並且在其基礎上結合 IdentityServer4 開發了一套 API 開放平臺。由於部分項目是基於 ABP 框架進行開發的,介面的平均 QPS 基本是在 2K~3K /S 左右 (E3 1231 16G)。採用 Ocelot 進行請 ...
  • 本文只涉及Helm的Chart操作,不會對其他知識進行過多描述。至於安裝這塊,麻煩自行百度吧,一大堆呢。 在容器化的時代,我們很多應用都可以部署在docker,很方便,而再進一步,我們還有工具可以對docker進行編排,Kubernetes就是一個很好的工具。再再進一步,Kubernetes出現了h ...
  • 紙殼CMS在設計上使用的是ASP.Net Core預設的IOC容器,通過依賴註入可以輕鬆替換掉原來的介面實現。例如在使用紙殼CMS做二次開發的過程中,可能要接入另一系統的用戶來作為CMS系統的用戶。這種情況下,可以不用修改原來的UserService,而是重新實現一個IUserService,然後用... ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...