[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
  • log4net:2.08

四、實現思路

4.1 切面實現

此處依賴自定義的日志組件,配置是否開啟調試模式,如果啟用調試模式,則會輸出請求參數信息以及響應參數信息。
代碼如下:

/// <summary>
/// 日志 攔截器
/// </summary>
public class LoggingInterceptor:IInterceptor
{
    /// <summary>
    /// 日志記錄器
    /// </summary>
    private static readonly ILog Logger = Log.GetLog(typeof(LoggingInterceptor));

    public void Intercept(IInvocation invocation)
    {
        try
        {                
            if (Logger.IsDebugEnabled)
            {
                Logger.Caption("日志攔截器-調試信息");
                Logger.Class(invocation.TargetType.FullName);
                Logger.Method(invocation.Method.Name);
                Logger.Params("參數:{0}", invocation.Arguments.ToJson());                    
            }
            invocation.Proceed();
            if (Logger.IsDebugEnabled)
            {
                if (invocation.ReturnValue != null && invocation.ReturnValue is IEnumerable)
                {
                    dynamic collection = invocation.ReturnValue;
                    Logger.Content("結果:行數:{0}", collection.Count);
                }
                else
                {
                    Logger.Content("結果:{0}", invocation.ReturnValue.ToJson());
                }
                Logger.Debug();
            }
        }
        catch (Exception e)
        {
            Logger.Caption("日志攔截器-異常");                
            Logger.Class(invocation.TargetType.FullName);
            Logger.Method(invocation.Method.Name);
            Logger.Params("參數:{0}", invocation.Arguments.ToJson());
            Logger.Exception(e);
            Logger.Error();
            throw;
        }
    }
}

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<LoggingInterceptor>();
        builder.RegisterAssemblyTypes(assembly)
            .Where(type => typeof(IDependency).IsAssignableFrom(type) && !type.GetTypeInfo().IsAbstract)
            .AsImplementedInterfaces()
            .InstancePerLifetimeScope()
            .EnableInterfaceInterceptors()
            .InterceptedBy(typeof(LoggingInterceptor));
    }
}

五、例子

輸出日志如下:

六、相關源碼

自定義日志組件可參考:JCE.DataCenter.Infrastructure
實現日志組件可參考:JCE.DataCenter.Logs


免責聲明!

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



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