Lind.DDD.UoW工作單元的實現

来源:http://www.cnblogs.com/lori/archive/2015/12/28/5083710.html
-Advertisement-
Play Games

回到目錄工作單元UoW我們幾乎在任務一個像樣的框架里都可以找到它的足跡,是的,對於大型系統來說,他是很重要的,保持數據一致性,維持事務狀態這都是它要為系統實現的功能,而在不同的框架里,實現UoW的機制也是不同的,在Lind.DDD中,採用了一種共同註冊,統一提交的方式來實現UoW!UoW結構圖我們來...


回到目錄

工作單元UoW我們幾乎在任務一個像樣的框架里都可以找到它的足跡,是的,對於大型系統來說,他是很重要的,保持數據一致性,維持事務狀態這都是它要為系統實現的功能,而在不同的框架里,實現UoW的機制也是不同的,在Lind.DDD中,採用了一種共同註冊,統一提交的方式來實現UoW!

UoW結構圖

我們來看一下大叔工作單元的代碼實現,首先看一下IUnitOfWorkRepository,我們的倉儲介面會實現它,以便之後我們的倉儲對象可以添加到工作單元里

    /// <summary>
    /// 工作單元中倉儲介面CRUD操作
    /// 需要使用工作單元的倉儲,需要實現本介面(IRepository,IExtensionRepository)
    /// </summary>
    public interface IUnitOfWorkRepository
    {
        /// <summary>
        /// 添加實體
        /// </summary>
        /// <param name="item"></param>
        void UoWInsert(IEntity item);
        /// <summary>
        /// 更新實體
        /// </summary>
        /// <param name="item"></param>
        void UoWUpdate(IEntity item);
        /// <summary>
        /// 移除實體
        /// </summary>
        /// <param name="item"></param>
        void UoWDelete(IEntity item);
    }

接下來,我們再來說一下IUnitOfWork介面,它是工作單元入口的介面,有添加到單元和提交單元兩個方法,使用簡單明瞭,內部有字典對象,用來存儲要提交的操作,這也是工作單元的核心,IEntity是實體的標識介面,所有實體都要繼承它,而IUnitOfWorkRepository是倉儲的標識介面,所以倉儲介面都要繼承它。

   /// <summary>
    /// 工作單元
    /// 所有數據上下文對象都應該繼承它,面向倉儲的上下文應該與具體實現(存儲介質,ORM)無關
    /// </summary>
    public interface IUnitOfWork
    {
        /// <summary>
        /// 向工作單元中註冊變更
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="entity"></param>
        /// <param name="type"></param>
        /// <param name="repository"></param>
        void RegisterChangeded(IEntity entity, SqlType type, IUnitOfWorkRepository repository);
        /// <summary>
        /// 向資料庫提交變更
        /// </summary>
        void Commit();
    }

我們來看一下,大叔是如何實現IUnitOfWork介面的吧

   /// <summary>
    /// 工作單元,主要用於管理事務性操作
    /// Author:Lind.zhang
    /// </summary>
    public class UnitOfWork : IUnitOfWork
    {
        #region Fields
        /// <summary>
        /// 操作行為字典
        /// </summary>
        private IDictionary<IEntity, IUnitOfWorkRepository> insertEntities;
        private IDictionary<IEntity, IUnitOfWorkRepository> updateEntities;
        private IDictionary<IEntity, IUnitOfWorkRepository> deleteEntities;

        #endregion

        #region Constructor

        public UnitOfWork()
        {
            insertEntities = new Dictionary<IEntity, IUnitOfWorkRepository>();
            updateEntities = new Dictionary<IEntity, IUnitOfWorkRepository>();
            deleteEntities = new Dictionary<IEntity, IUnitOfWorkRepository>();
        }

        #endregion

        #region IUnitOfWork 成員
        /// <summary>
        /// 事務提交
        /// </summary>
        public void Commit()
        {
            try
            {
                using (TransactionScope transactionScope = new TransactionScope())
                {

                    foreach (var entity in insertEntities.Keys)
                    {
                        insertEntities[entity].UoWInsert(entity);
                    }
                    foreach (var entity in updateEntities.Keys)
                    {
                        updateEntities[entity].UoWUpdate(entity);
                    }
                    foreach (var entity in deleteEntities.Keys)
                    {
                        deleteEntities[entity].UoWDelete(entity);
                    }
                    transactionScope.Complete();//提交事務,程式中如果出錯,這行無法執行,即事務不會被提交,這就類似於rollback機制
                }
            }
            catch (Exception ex)
            {
                Logger.LoggerFactory.Instance.Logger_Error(ex);
            }

        }



        /// <summary>
        /// 註冊數據變更
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="entity"></param>
        /// <param name="type"></param>
        public void RegisterChangeded(IEntity entity, SqlType type, IUnitOfWorkRepository repository)
        {
            switch (type)
            {
                case SqlType.Insert:
                    insertEntities.Add(entity, repository);
                    break;
                case SqlType.Update:
                    updateEntities.Add(entity, repository);
                    break;
                case SqlType.Delete:
                    deleteEntities.Add(entity, repository);
                    break;
                default:
                    throw new ArgumentException("you enter reference is error.");
            }
        }
        #endregion


    }

工作單元在調用時也是非常方便的,兩步完成,第一註意動作,第二提交事務,下麵看一下DEMO的代碼片斷

        unitOfWork.RegisterChangeded(entity, SqlType.Update, userRepository);
            var userExtension = userExtRepository.Find(entity.Id);
            userExtension.NickName = Request.Form["UserExtension.NickName"];
            userExtension.School = Request.Form["UserExtension.School"];
            unitOfWork.RegisterChangeded(userExtension, SqlType.Update, userExtRepository);
            unitOfWork.Commit();

OK,對於工作單元的探討今天就先說到這裡,以後肯定還是機會去研究的!

回到目錄


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

-Advertisement-
Play Games
更多相關文章
  • 你可以從你們現在項目裡面隨便找幾處註釋,看看寫註釋的代碼是不是存在如下兩種毛病之一:1. 命名不准確;2. 方法太長(超過50行)。如果你找到的代碼沒有出現上面兩種毛病而註釋依然存在,那你再看看這個註釋是否有實際意義,是不是這個註釋不要也無所謂呢。註釋是惡魔這個觀點可能你第一次看到,你可能很難接受,...
  • 一、 生成對象的原始模式假定我們把Bob看成一個對象,它有"name"和"age"兩個屬性。 var person = { name : '', age : }現在,我們需要根據這個原型對象的規格(schema),生成兩個實例對象。 var person1 = {}; // 創建...
  • 編者按:本文由PMCAFF產品經理社區原創專欄作者 喬向陽 翻譯自 http://goodui.org/一個好的界面設計應該擁有高轉化率且方便用戶使用,換句話說:既能達到商業目的又能滿足方便易用的要求。有一個設計咨詢公司根據自己的客戶案例,總結了 75 個經過實踐證明的原則:之前國內流傳這篇文章的前...
  • Reactor事件驅動的兩種設計實現:面向對象 VS 函數式編程這裡的函數式編程的設計以muduo為例進行對比說明;Reactor實現架構對比面向對象的設計類圖如下:函數式編程以muduo為例,設計類圖如下:面向對象的Reactor方案設計我們先看看面向對象的設計方案,想想為什麼這麼做; 拿出Rea...
  • 最近的工作比較清閑,公司目前想開展一個財富類的P2B的一個門戶,另外碰上移動公司核心測試,元旦後要飛深圳出差,所以這個交叉點工作安排下去了,自己比較清閑一些,我也把自己開展南京生活這個網站的歷程進行記錄下來。想要開展移動互聯網創業,就要有自己的門戶,怎麼把這個門戶弄起來,確實需要慎重思考。但是也不需...
  • 回到目錄Lind.DDD框架里提出了對數據集的控制,某些許可權的用戶為某些表添加某些數據集的許可權,具體實現是在一張表中存儲用戶ID,表名,檢索欄位,檢索值和檢索操作符,然後用戶登陸後,通過自己許可權來構建對應表的查詢語句,即動態構建表達式樹,這種操作一些被寫在業務層上,我們可以在業務層需要進行數據集許可權...
  • 1.橋接模式 個人覺得有些類似於抽象工廠模式2.命令模式 提供了一個中介,中介集成了各類功能,客戶端可以添加刪除某些功能,可以執行某些功能
  • 一些大型網站常用到的技術概覽,經驗談。
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...