手工搭建基於ABP的框架 - 工作單元以及事務管理

来源:https://www.cnblogs.com/skabyy/archive/2018/02/11/8440439.html
-Advertisement-
Play Games

一個業務功能往往不只由一次資料庫請求(或者服務調用)實現。為了功能的完整性,我們希望如果該功能執行一半時出錯,則撤銷前面已執行的改動。在資料庫層面上,事務管理實現了這種完整性需求。在ABP中,一個完整的業務功能稱為一個工作單元(Unit of Work,簡稱UoW)。工作單元代表一種完整的、原子性的 ...


一個業務功能往往不只由一次資料庫請求(或者服務調用)實現。為了功能的完整性,我們希望如果該功能執行一半時出錯,則撤銷前面已執行的改動。在資料庫層面上,事務管理實現了這種完整性需求。在ABP中,一個完整的業務功能稱為一個工作單元(Unit of Work,簡稱UoW)。工作單元代表一種完整的、原子性的操作。即一個工作單元包含的步驟要麼全部被執行,要麼都不被執行。如果執行一半時出現異常,則必須講已執行的步驟還原。通常我們將事務管理實現在工作單元中。下麵我們從ABP源碼入手研究如何使用工作單元。

ABP工作單元(UoW)的工作原理

ABP預設將工作單元應用在Repositories、 Application Services、MVC控制器和Web API控制器等組件。也就是說,這些組件的每個方法都是一個工作單元。ABP文檔對工作單元的原理講得不是很詳細,所以我們只能通過源碼進行研究。這裡我們以MVC控制器為例來瞭解一下ABP工作單元大致的工作原理。源碼分析比較枯燥,最好配套ABP源碼閱讀,或者跳到後面看粗體字結論

ABP在Web模塊初始化時註冊了過濾器AbpMvcUowFilterAbpMvcUowFilter在請求處理前(OnActionExecuting方法)調用UnitOfWorkManager.Begin方法來開始一個工作單元。UnitOfWorkManager.Begin創建一個IUnitOfWork的實例並賦值給ICurrentUnitOfWorkProvider.Current,然後調用IUnitOfWork.Begin方法開始一個工作單元。在請求處理結束後(OnActionExecuted方法)如果處理過程沒有異常就調用IUnitOfWork.Complete方法完成工作單元,並且無論請求處理是否成功,都調用IUnitOfWork.Dispose來結束工作單元。

ABP提供了一個實現IUnitOfWork的抽象基類UnitOfWorkBase,另外還有個繼承了UnitOfWorkBase的類NullUnitOfWorkNullUnitOfWork定義上面有一段註釋如此寫到:

/// <summary>
/// Null implementation of unit of work.
/// It's used if no component registered for <see cref="IUnitOfWork"/>.
/// This ensures working ABP without a database.
/// </summary>
public sealed class NullUnitOfWork : UnitOfWorkBase

NullUnitOfWork是一個“空”的工作單元,它不會做任何操作。如果我們沒有在IoC容器中註冊其它IUnitOfWork的實現類,則ABP預設使用不做任何事的NullUnitOfWork作為工作單元。所以如果我們要做一些保證功能完整性的工作(比如開啟資料庫事務),就要實現IUnitOfWork並註冊到IoC容器

閱讀UnitOfWorkBase可以看到,UnitOfWorkBase分別在Begin方法、Complete方法和Dispose方法中調用了BeginUow方法、CompleteUow方法和DisposeUow方法。我們需要重寫的主要是BeginUowCompleteUowDisposeUow這三個方法

通過源碼簡單瞭解了原理後,我們後面寫代碼要註意的有下麵幾點:

  1. 寫一個繼承UnitOfWorkBase的類UnitOfWork,並實現介面ITransientDependency保證UnitOfWork被註冊到IoC容器
  2. 重寫方法UnitOfWorkBase.BeginUow,實現工作單元開始時的啟動操作
  3. 重寫方法UnitOfWorkBase.CompleteUow,實現工作單元正常結束時的保存操作
  4. 重寫方法UnitOfWorkBase.DisposeUow,實現工作單元結束時的清理操作
  5. 通過ICurrentUnitOfWorkProvider.Current來獲取當前的工作單元

重寫SessionProvider,並實現工作單元

在之前文章(手工搭建基於ABP的框架(2) - 訪問資料庫)實現的LocalDbSessionProvider中,為了追求代碼簡單,我們粗暴地用一個實質上是全局的變數來保存資料庫Session,在每次訪問資料庫時,flush上一個Session並創建新Session。另一方面,資料庫連接的配置、Session的創建保存、以及Session的提供都胡亂地放在了這個類里。這其實是非常不合理而且會引發很多問題的實現方法。

下麵我們重新設計這一塊邏輯。我們將LocalDbSessionProvider所負責的功能拆分,分別實現在LocalDbSessionConfigurationUnitOfWorkUnitOfWorkLocalDbSessionProvider三個類中:

  • LocalDbSessionConfiguration,單例。實現資料庫連接配置,提供資料庫Session工廠。

    public class LocalDbSessionConfiguration : ILocalDbSessionConfiguration, IDisposable
    {
        protected FluentConfiguration FluentConfiguration { get; private set; }
    
        public ISessionFactory SessionFactory { get; }
    
        public LocalDbSessionConfiguration()
        {
            FluentConfiguration = Fluently.Configure();
            // 資料庫連接串
            var connString = "data source=|DataDirectory|MySQLite.db;";
            FluentConfiguration
                // 配置連接串
                .Database(SQLiteConfiguration.Standard.ConnectionString(connString))
                // 配置ORM
                .Mappings(m => m.FluentMappings.AddFromAssembly(Assembly.GetExecutingAssembly()));
            // 生成session factory
            SessionFactory = FluentConfiguration.BuildSessionFactory();
        }
    
        public void Dispose()
        {
            SessionFactory.Dispose();
        }
    }
  • UnitOfWork,管理資料庫Session的生命期。資料庫Session和事務的創建、銷毀都封裝在這裡。這裡的一個問題是何時創建資料庫Session。一個自然的想法是在BeginUow創建。然而這並不是一個很好的方式,會產生如下問題:1、預設情況下,Controllers、Application Services和Repositories都會開啟工作單元,也就是說,一次HTTP請求可能會多次開啟工作單元,導致過多地創建資料庫Session,甚至導致資料庫被鎖;2、即使某個介面不需要訪問資料庫,工作單元仍然會創建資料庫Session,浪費資源。正確的做法是在需要獲取資料庫Session的時候才進行創建。在下麵的實現中,我們將在UnitOfWork實現一個GetOrCreateSession方法來獲取資料庫Session。該方法在第一次調用時創建一個資料庫Session並開啟事務,後續調用則返回前面已創建的資料庫Session。後面UnitOfWorkLocalDbSessionProvider將調用這個方法來獲取資料庫Session。

    public class UnitOfWork : UnitOfWorkBase, ITransientDependency
    {
        public ILocalDbSessionConfiguration DbSessionConfiguration { get; }
    
        private ISession _session;
    
        public UnitOfWork(
            IConnectionStringResolver connectionStringResolver,
            IUnitOfWorkDefaultOptions defaultOptions,
            IUnitOfWorkFilterExecuter filterExecuter,
            ILocalDbSessionConfiguration localDbSessionConfiguration)
            : base(connectionStringResolver, defaultOptions, filterExecuter)
        {
            DbSessionConfiguration = localDbSessionConfiguration;
        }
    
        public ISession GetOrCreateSession()
        {
            if (_session == null)
            {
                _session = DbSessionConfiguration.SessionFactory.OpenSession();
                _session.BeginTransaction();
            }
            return _session;
        }
    
        public override void SaveChanges()
        {
            _session?.Flush();
        }
    
        public override Task SaveChangesAsync()
        {
            // 我們不用非同步Action,就不實現這個方法了。
            throw new NotImplementedException();
        }
    
        protected override void CompleteUow()
        {
            SaveChanges();
            _session?.Transaction?.Commit();
        }
    
        protected override Task CompleteUowAsync()
        {
            // 我們不用非同步Action,就不實現這個方法了。
            throw new NotImplementedException();
        }
    
        protected override void DisposeUow()
        {
            _session?.Transaction?.Dispose();
            _session?.Dispose();
        }
    }
    
    internal static class UnitOfWorkExtensions
    {
        public static ISession GetSession(this IActiveUnitOfWork unitOfWork)
        {
            if (unitOfWork == null)
            {
                throw new ArgumentNullException(nameof(unitOfWork));
            }
    
            if (!(unitOfWork is UnitOfWork))
            {
                throw new ArgumentException("unitOfWork is not type of " + typeof(UnitOfWork).FullName, nameof(unitOfWork));
            }
    
            return (unitOfWork as UnitOfWork).GetOrCreateSession();
        }
    }
  • UnitOfWorkLocalDbSessionProvider,單例。通過當前的工作單元來提供資料庫Session。

    public class UnitOfWorkLocalDbSessionProvider : ISessionProvider, ISingletonDependency
    {
        private readonly ICurrentUnitOfWorkProvider _unitOfWorkProvider;
    
        public UnitOfWorkLocalDbSessionProvider(ICurrentUnitOfWorkProvider currentUnitOfWorkProvider)
        {
            _unitOfWorkProvider = currentUnitOfWorkProvider;
        }
    
        public ISession Session => _unitOfWorkProvider.Current?.GetSession();
    }

最後,TweetRepositoryTweetQueryService的構造函數用到了舊的LocalDbSessionProvider,這兩處也需要改一下:

public TweetRepository()
    : base(IocManager.Instance.Resolve<UnitOfWorkLocalDbSessionProvider>())
{ }
public TweetQueryService()
    : base(IocManager.Instance.Resolve<UnitOfWorkLocalDbSessionProvider>())
{ }

使用NHProfiler進行驗證

上面實現了工作單元並封裝了資料庫事務管理。我們需要有方法驗證資料庫訪問時確實開啟了事務。NHProfiler是一個能夠監視NHibernate生成的SQL語句的工具。我們將使用NHProfiler查看生成的SQL,確認實現了工作單元後確實開啟了事務管理。

NHProfiler由兩個部分組成:

  1. 一個嵌入到我們應用的DLL。這個DLL會在NHibernate訪問資料庫時往本地socket發送生成的SQL語句。
  2. 客戶端。這個客戶端通過socket接收上面所說的DLL發送的數據並展示。

接下來我們的程式需要做一些小改動。首先MyTweet.Web項目需要引用NHProfiler包里的HibernatingRhinos.Profiler.Appender.dll。或者從Nuget添加NHibernateProfiler.Appender包。如果你從NuGet添加的,則需要確認NuGet包的版本和客戶端的版本一致。

添加引用後,我們還需要在入口函數Application_Start加上HibernatingRhinos.Profiler.Appender.NHibernate.NHibernateProfiler.Initialize()這句語句來開啟NHProfiler的監聽:

protected override void Application_Start(object sender, EventArgs e)
{
    AreaRegistration.RegisterAllAreas();
    FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
    RouteConfig.RegisterRoutes(RouteTable.Routes);
    BundleConfig.RegisterBundles(BundleTable.Bundles);

    HibernatingRhinos.Profiler.Appender.NHibernate.NHibernateProfiler.Initialize();

    IocManager.Instance.IocContainer.AddFacility<LoggingFacility>(
        f => f.UseAbpLog4Net().WithConfig("log4net.config"));

    base.Application_Start(sender, e);
}

現在我們啟動MyTweet.Web應用,然後打開NHProfiler客戶端。這時NHProfiler已經在監聽NHibernate生成的SQL了。到我們的應用新建一條tweet,再回到NHProfiler客戶端,可以看到,新建tweet的操作確實被事務包裹起來了。

總結

在本文中,我們自己繼承UnitOfWorkBase實現工作單元,使得整個框架更靈活,更容易擴展。在現有代碼上稍作修改,我們還可以支持多資料庫事務。


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

-Advertisement-
Play Games
更多相關文章
  • 參考於 http://jinnianshilongnian.iteye.com/blog/2186448 1、example.conf配置文件 location ~ /lua_request/(\d+)/(\d+) { #設置nginx變數 set $a $1; set $b $host; defa ...
  • 一、給 tomcat 添加用戶名、密碼 在安裝目錄/cong/tomcat-user.xml中添加以下配置 二、在 maven 項目中添加 tomcat 插件 1. 在 pom.xml 中添加插件和依賴 三、startup.bat 啟動伺服器 再右擊項目>run as>maven build... ...
  • 扯淡 祝各位在園裡的朋友新年快樂! 辛苦一年,為更好的自己也為更好的世界,很多人要感謝你們,你們也應該有很多人要感謝吧。 看了馬斯克的採訪視頻,又想起 "蘭迪·鮑許的最後一課" ,時光遷移,唯有夢想可堅持。 概念 在java的世界里,基於jvm實現的語言最終要進入jvm編譯的流程都需要把上層高級語言 ...
  • 在應用中使用 SQLite 資料庫來存儲數據是相當常見的。在 UWP 平臺中要使用 SQLite,一般會使用 SQLite for Universal Windows Platform 和 SQLite PCL 之類的庫,前者是 SQLite 引擎庫,而後者則提供了用於操作資料庫的 API ,不過自 ...
  • 需要在派生類實現lEntity,IEntity的兩個屬性EntityId和IsDeleteInDataBase,以顯式方式實現 Model的所有屬性存入資料庫以前都要確保有相應的預設值,不要依賴資料庫的預設值,建議在實體中增加New()方法用於添加實體前創建實體,例如: [PetaPoco.Tabl ...
  • 在C#的繼承中嘗嘗會用到相關的修飾詞:override和new。這兩個修飾符都可以在新的子類中,重寫同名的父類方法。 override: 擴展或修改繼承的方法、屬性、索引器或事件的抽象或虛擬實現需要使用到。 new:在用作聲明修飾符時,new 關鍵字可以顯式隱藏從基類繼承的成員。 隱藏繼承的成員時, ...
  • 1.下載ABP項目模板, 打開網址https://aspnetboilerplate.com/Templates,選擇MultiPage Web Application,輸入項目名稱和驗證碼,即可點擊Create My Project按鈕下載代碼 2.下載完成之後,會跳到https://aspnet ...
  • 支持:可指定絕對過期時間、滑動過期明間、文件依賴 三種緩存方式,目前已在公司各種生產業務項目中有使用。優點是可以根據數據的使用頻率設置緩存有效期,特別是文件依賴緩存,比如:連接字元串讀取一次後,若CONFIG文件沒有改變,則緩存永久有效,一旦CONFIG更改,則緩存失效需重新讀取,保證數據緩存的最大 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...