淺談 EF CORE 遷移和實例化的幾種方式

来源:http://www.cnblogs.com/Wddpct/archive/2017/07/28/7249670.html
-Advertisement-
Play Games

出於學習和測試的簡單需要,使用 Console 來作為 EF CORE 的承載程式是最合適不過的。今天筆者就將平時的幾種使用方式總結成文,以供參考,同時也是給本人一個溫故知新的機會。因為沒有一個完整的脈絡,所以也只是想起什麼寫點什麼,不通順的地方還請多多諒解。 本文對象資料庫預設為 VS 自帶的 L ...


出於學習和測試的簡單需要,使用 Console 來作為 EF CORE 的承載程式是最合適不過的。今天筆者就將平時的幾種使用方式總結成文,以供參考,同時也是給本人一個溫故知新的機會。因為沒有一個完整的脈絡,所以也只是想起什麼寫點什麼,不通順的地方還請多多諒解。

本文對象資料庫預設為 VS 自帶的 LocalDB

1. Normal & Simple

先介紹一種最簡單的構建方式,人人都會。

  • 新建 Console 應用程式,命名自定

  • 安裝相關Nuget 包

//Sql Server Database Provider
Install-Package Microsoft.EntityFrameworkCore.SqlServer
    
//提供熟悉的Add-Migration,Update-Database等Powershell命令,不區分關係型資料庫類型
Install-Package Microsoft.EntityFrameworkCore.Tools 
  • 自定義 DbContext
public class MyContext:DbContext
{
    protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
    {
        optionsBuilder.UseSqlServer("Server=(localdb)\\mssqllocaldb;Database=ConsoleApp;Trusted_Connection=True;MultipleActiveResultSets=true;");
    }
}
  • 執行遷移和更新命令
Add-Migration Initialize
Update-Database
  • 使用方式
using (var context = new MyContext())
{
    // TODO
}

剛以上,我們便見識到了了一種最平常也是最簡單的使用方式,接下來,讓我們用其他方式去慢慢地改造它,從而儘可能地接觸更多的用法。

2. Level Up

2.1 準備工作

將第一步生成的資料庫,遷移文件和使用方式內容全部刪除。

2.2 更新 MyContext 內容

刪除 MyContext 中的 OnConfiguring 方法及其內容,增加含有 DbContextOptions 類型參數的構造器,我們的MyContext看起來應該是下麵這個樣子。

public class MyContext : DbContext
{
    public MyContext(DbContextOptions options) : base(options)
    {
    }
}

假如我們此時仍然再執行遷移命令,VS將提示以下錯誤

No parameterless constructor was found on 'MyContext'. Either add a parameterless constructor to 'MyContext' or add an implementation of 'IDbContextFactory' in the same assembly as 'MyContext'.

添加無參構造器的方式之後再講解,先來按照提示信息添加一個 IDbContextFactory 的實現類。

public class MyContextFactory : IDbContextFactory<MyContext>
{
    public MyContext Create(DbContextFactoryOptions options)
    {
        var optionsBuilder = new DbContextOptionsBuilder<MyContext>();
        optionsBuilder.UseSqlServer("Server=(localdb)\\mssqllocaldb;Database=ConsoleApp;Trusted_Connection=True;MultipleActiveResultSets=true;");
            
        return new MyContext(optionsBuilder.Options);
    }
}

之後再次運行遷移和更新資料庫的命令也是水到渠成。

2.3 使用方式:構造器實例化

既然 MyContext 含有 DbContextOptions 類型參數的構造器,那就手動創建一個參數實例註入即可。

var contextOptionsBuilder = new DbContextOptionsBuilder<MyContext>();
contextOptionsBuilder.UseSqlServer("Server=(localdb)\\mssqllocaldb;Database=ConsoleApp;Trusted_Connection=True;MultipleActiveResultSets=true;");

// 註入配置選項
using (var context = new MyContext(contextOptionsBuilder.Options))
{
    // TODO
}

經此,我們知道了遷移命令會檢測 Context 的相關配置入口,只有在滿足存在 OnConfiguring 方法或者存在自建 IDbContextFactory 實現類的情況下,命令才能成功運行。

3. Day Day Up

目前為止,我們已經知道如何手動遷移和實例化 Context 的步驟了所以讓我們更進一步。寫過 ASP.NET CORE 的人可能知道在 ASP.NET CORE 中,Context 常常以依賴註入的方式引入到我們的 Web 層,Service 層,或者 XXCore 層中(話說筆者最近最喜歡的解決方案開發架構就是偽 DDD 的四層架構,有空再介紹吧)。其實在 Console 應用中,這也可以很容易實現,具體的依賴註入引入可以參考筆者的上一篇博客,所以最終的代碼效果如下:

var serviceCollection = new ServiceCollection();
serviceCollection.AddDbContext<MyContext>(c =>
{
    c.UseSqlServer("Server=(localdb)\\mssqllocaldb;Database=ConsoleApp;Trusted_Connection=True;MultipleActiveResultSets=true;");
});

var serviceProvider = serviceCollection.BuildServiceProvider();

using (var context = serviceProvider.GetService<MyContext>())
{
    //context.Database.Migrate();
}

至此,我們便基本完成了本文的主題,唯一有些美中不足的是我們的資料庫連接字元串好像到處都是,這不是什麼大問題,筆者直接將 Configuration 的配置代碼貼在下麵,這也是 ABP 中的方式。

public class AppConfigurations
{
    private static readonly ConcurrentDictionary<string, IConfigurationRoot> ConfigurationCache;

    static AppConfigurations()
    {
        ConfigurationCache = new ConcurrentDictionary<string, IConfigurationRoot>();
    }

    public static IConfigurationRoot Get(string environmentName = null)
    {
        var cacheKey = "#" + environmentName;
        return ConfigurationCache.GetOrAdd(
            cacheKey,
            _ => BuildConfiguration(environmentName)
        );
    }

    private static IConfigurationRoot BuildConfiguration(string environmentName = null)
    {
        var builder = new ConfigurationBuilder()
            .SetBasePath(Directory.GetCurrentDirectory())
            .AddJsonFile("appsettings.json", true, true);

        if (!string.IsNullOrWhiteSpace(environmentName))
            builder = builder.AddJsonFile($"appsettings.{environmentName}.json", true);

        builder = builder.AddEnvironmentVariables();

        return builder.Build();
    }
}

這個工具類的使用方式就不再贅述了。

4. 結尾

最後,想必會有人問為什麼要折騰這樣一個小小的 Console 應用呢?其實通過這樣一步步下來,我們可以發現一些項目功能上的亮點,比如既然可以自配置 DbContext 的 Option 選項,同時我們也知道瞭如何在類庫和 Console 項目中添加依賴註入以及 Configuration 提取鏈接參數的功能,那針對三層架構或是 DDD 項目增加含真實資料庫或是記憶體資料庫(InMemory)的單元測試,或者是自動Migrate Context 和更新資料庫也將是十分簡單的一件事,至少看起來會比官方的示例更加真實和具有可操作性。而這部分內容筆者也將會在之後的博文中給出。


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

-Advertisement-
Play Games
更多相關文章
  • 最近新上線了一個網站,專門收集網上簽到賺錢,有獎活動等等的網站 我就要集分寶 http://www.591jfb.com。新建立 了一個欄目“每日更新”,這樣就需要每天都登錄到網站後臺去發佈文章,感覺有些繁瑣,於是就想找點省勁的辦法,於是便有了此文。 搜索下載了蘇飛提供的httphelper,比著例 ...
  • Owin Startup 類解析 每個 Owin 程式都有 startup 類,在這個 startup 類裡面你可以指定應用程式管道模型中的組件。你可以通過不同的方式來連接你的 startup 類和運行時,這些取決於你選擇的宿主模型(OwinHost, IIS, and IIS-Express)。 ...
  • C#Socket 網路通信非同步處理 SocketAsyncEventArgs 非同步套接字操作 1.服務端簡單實現: 2.客戶端簡單實現: ...
  • 背景:以前做登錄時用的都是FormsAuthentication.SetAuthCookie(model.UID, IsRemeber),但是有一個不好,不能存儲多個值,有時候我們既想存儲登錄用戶的UID又想存儲用戶名,以前都是將兩者拼接成字元串,用的時候在split出來,比較麻煩,現在用Claim ...
  • 首先,我們需要做的是什麼? 我們成功登錄之後,跳轉到主界面,然後主界面的登錄按鈕變成頭像啥的。下一次打開網頁就要判斷有沒有登錄過,有cookie就不需要登錄,直接顯示頭像 1.成功登錄後,客戶端請求伺服器 2.把登陸信息傳入到伺服器 3.伺服器有了這個cookie,保存到cookie集合裡面,然後反 ...
  • 出現這個原因我們應該都能猜測到,文件傳出過大,超出了WCF預設範圍,那麼我們需要進行修改。 服務端和客戶端都需要修改。 第一、客戶端: 上面是wcf客戶端預設生成的,如需要修改傳輸文件大小,需要一下改動 <binding name="BasicHttpBinding_ITaskService" ma ...
  • IIS 中的 Owin 在 IIS 裡面部署 Owin,既能得到 Owin 管道模型的靈活性和模塊特性,也能很好地利用 IIS 成熟的配置,Owin 程式將會跑在 ASP.NET request 的管道中。 首先建一個空的 Web 項目 添加 Nuget 包 Microsoft.Owin.Host. ...
  • 一、序言 大家或多或少都聽過WebService(Web服務),有一段時間很多電腦期刊、書籍和網站都大肆的提及和宣傳WebService技術,其中不乏很多吹噓和做廣告的成分。但是不得不承認的是WebService真的是一門新興和有前途的技術,那麼WebService到底是什麼?何時應該用? 當前的 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...