[AOP系列]Autofac+Castle實現AOP事務


一、前言

最近公司新項目,需要搭架構進行開發,其中需要保證事務的一致性,經過一番查找,發現很多博文都是通過Spring.Net、Unity、PostSharp、Castle Windsor這些方式實現AOP的。但是這不是我想要的,因此一番查找后,使用Autofac、DynamicProxy該方式實現AOP。

二、使用AOP的優勢

博主覺得它的優勢主要表現在:

  • 將通用功能從業務邏輯中抽離出來,就可以省略大量重復代碼,有利於代碼的操作和維護。
  • 在軟件設計時,抽出通用功能(切面),有利於軟件設計的模塊化,降低軟件架構的復雜程度。也就是說通用的功能就是一個單獨的模塊,在項目的主業務里面是看不到這些通用功能的設計代碼的。

三、引用庫

  • Autofac:4.6
  • Autofac.Extras.DynamicProxy:4.1.0
  • Castle.Core:3.2.2

四、實現思路

4.1 定義屬性

定義屬性,通過當前方法是否包含該屬性進行判斷開啟事務,如果存在該屬性則開啟事務,否則忽略事務。
事務屬性可以設置超時時間、事務范圍以及事務隔離級別。
代碼如下:

/// <summary>
/// 開啟事務屬性
/// </summary>
[AttributeUsage(AttributeTargets.Method,Inherited = true)]
public class TransactionCallHandlerAttribute:Attribute
{
    /// <summary>
    /// 超時時間
    /// </summary>
    public int Timeout { get; set; }

    /// <summary>
    /// 事務范圍
    /// </summary>
    public TransactionScopeOption ScopeOption { get; set; }

    /// <summary>
    /// 事務隔離級別
    /// </summary>
    public IsolationLevel IsolationLevel { get; set; }

    public TransactionCallHandlerAttribute()
    {
        Timeout = 60;
        ScopeOption=TransactionScopeOption.Required;
        IsolationLevel=IsolationLevel.ReadCommitted;
    }
}

4.2 切面實現

獲取當前方法是否包含TransactionCallHandlerAttribute該屬性,如果有該屬性則開啟事務。
本人在此處加入開發模式判斷,用於沒設置MSDTC產生異常的問題,如果不需要可忽略。
另外日志功能自行實現即可。
代碼如下:

/// <summary>
/// 事務 攔截器
/// </summary>
public class TransactionInterceptor:IInterceptor
{
    //可自行實現日志器,此處可忽略
    /// <summary>
    /// 日志記錄器
    /// </summary>
    private static readonly ILog Logger = Log.GetLog(typeof(TransactionInterceptor));

    // 是否開發模式
    private bool isDev = false;
    public void Intercept(IInvocation invocation)
    {
        if (!isDev)
        {
            MethodInfo methodInfo = invocation.MethodInvocationTarget;
            if (methodInfo == null)
            {
                methodInfo = invocation.Method;
            }
                            
            TransactionCallHandlerAttribute transaction =
                methodInfo.GetCustomAttributes<TransactionCallHandlerAttribute>(true).FirstOrDefault();
            if (transaction != null)
            {
                TransactionOptions transactionOptions = new TransactionOptions();
                //設置事務隔離級別
                transactionOptions.IsolationLevel = transaction.IsolationLevel;
                //設置事務超時時間為60秒
                transactionOptions.Timeout = new TimeSpan(0, 0, transaction.Timeout);
                using (TransactionScope scope = new TransactionScope(transaction.ScopeOption, transactionOptions))
                {
                    try
                    {
                        //實現事務性工作
                        invocation.Proceed();
                        scope.Complete();
                    }
                    catch (Exception ex)
                    {
                        // 記錄異常
                        throw ex;
                    }
                }
            }
            else
            {
                // 沒有事務時直接執行方法
                invocation.Proceed();
            }
        }
        else
        {
            // 開發模式直接跳過攔截
            invocation.Proceed();
        }
    }
}

4.3 切面注入

博主對Autofac進行了封裝,可能與你們的配置不一樣,但是,Load(ContainerBuilder builder)該方法內容是一致的,因此注入方式一致的。
通過定義IDependency空接口方式,需要注入的類則繼承該接口即可。
代碼如下:

/// <summary>
/// 應用程序IOC配置
/// </summary>
public class IocConfig : ConfigBase
{
    // 重寫加載配置
    protected override void Load(ContainerBuilder builder)
    {
        var assembly = this.GetType().GetTypeInfo().Assembly;
        builder.RegisterType<TransactionInterceptor>();
        builder.RegisterAssemblyTypes(assembly)
            .Where(type => typeof(IDependency).IsAssignableFrom(type) && !type.GetTypeInfo().IsAbstract)
            .AsImplementedInterfaces()
            .InstancePerLifetimeScope()
            .EnableInterfaceInterceptors()
            .InterceptedBy(typeof(TransactionInterceptor));
    }
}

五、例子

/// <summary>
/// 添加文章
/// </summary>
/// <param name="name"></param>
[TransactionCallHandler]
public void AddArticle(string name)
{
    BasArticle model=new BasArticle();
    model.ArticleID = Guid.Empty;//故意重復,判斷是否會回滾。
    model.Code = TimestampId.GetInstance().GetId();
    model.Name = name;
    model.Status = 1;
    model.Creater = "測試";
    model.Editor = "測試";
    this._basArticleRepository.Insert(model);            
}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM