一、前言 最近公司新項目,需要搭架構進行開發,其中需要對一些日誌進行輸出,經過一番查找,發現很多博文都是通過Spring.Net、Unity、PostSharp、Castle Windsor這些方式實現AOP的。但是這不是我想要的,因此一番查找後,使用 該方式實現AOP。 二、使用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