AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点……是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
动态代理是实现AOP的一种方式,即在开发过程中我们不需要处理切面中(日志等)的工作,而是在运行时,通过动态代理来自动完成。Castle DynamicProxy是一个实现动态代理的框架,被很多优秀的项目用来实现AOP编程,EF Core、Autofac等。
为业务类添加AOP拦截器
- ///
- /// 为业务类添加AOP拦截器。
- ///
- public class InterceptorAttribute:PlutoStudio.Aop.InterceptorAttribute
- {
- ///
- /// 拦截方法的执行,如果当前方法有拦截处理器,则执行处理器。
- ///
- /// name"invocation">被拦截的调用目标对象
- public override void Intercept(IInvocation invocation)
- {
- var method = invocation.Method;
- var processors = method.GetCustomAttributes(typeof(IInterceptorProcessor),true).Cast
().ToList(); - processors.ForEach(p => PlutoStudio.MefContainer.Container.ComposeParts(p));
- if (processors.Count>0)
- {
- processors.ForEach(p => p.PreProcess(invocation));
- try
- {
- invocation.Proceed();
- processors.ForEach(p => p.PostProcess(invocation, null));
- }
- catch (Exception ex)
- {
- processors.ForEach(p => p.PostProcess(invocation, ex));
- throw;
- }
- }
- else
- {
- invocation.Proceed();
- }
- }
- }
- ///
- /// 拦截器处理器接口。
- ///
- public interface IInterceptorProcessor
- {
- ///
- /// 拦截器处理方法,在目标方法执行前执行。
- ///
- /// name"invocation">拦截的目标对象
- void PreProcess(IInvocation invocation);
- ///
- /// 拦截器处理方法,在目标方法执行后执行。
- ///
- /// name"invocation">拦截的目标对象
- /// name"ex">目标方法的异常
- void PostProcess(IInvocation invocation,Exception ex);
- }
日志处理器
可以将目标方法的信息保存到日志系统中
- ///
- /// 日志处理器,可以将目标方法的信息保存到日志系统中。
- ///
- [AttributeUsage(AttributeTargets.Method, AllowMultiple = false, Inherited = true)]
- public class LogAttribute : Attribute, IInterceptorProcessor
- {
- [Import(AllowDefault = true)]
- public ILog Log { get; set; }
- ///
- /// 在目标方法执行完成后执行,这里会记录目标方法的相关信息并写入日志系统。
- ///
- /// name"invocation">
- /// name"ex">
- public void PostProcess(IInvocation invocation, Exception ex)
- {
- if (Log != null)
- {
- var @class = invocation.TargetType.FullName;
- var method = invocation.Method.Name;
- var parameterNames = invocation.Method.GetParameters().Select(p => p.Name).ToList();
- var args = invocation.Arguments;
- var parameters = new Dictionary
(); - for (int i = 0; i < parameterNames.Count; i++)
- {
- parameters.Add(parameterNames[i], args[i]);
- }
- var returnValue = invocation.ReturnValue;
- var stackTrace = new StackTrace(true);
- var stacks = stackTrace.GetFrames();
- var stack = stacks.SkipWhile(i => i.GetMethod().Name != invocation.Method.Name).Select(GetStack);
- var log = new TraceLog
- {
- Class = @class,
- Method = method,
- Parameter = parameters,
- ReturnValue = returnValue,
- Strack = stack,
- Exception = ex,
- };
- Log.Custom(log);
- }
- }
- private object GetStack(StackFrame frame)
- {
- var method = frame.GetMethod();
- var type = method.ReflectedType;
- if (type.FullName.StartsWith("Castle.Proxies"))
- {
- type = type.BaseType;
- }
- return new
- {
- Method = method.Name,
- Type = type.FullName,
- File = frame.GetFileName(),
- Line = frame.GetFileLineNumber(),
- };
- }
- public void PreProcess(IInvocation invocation)
- {
- }
- }
- ///
- /// 系统跟踪日志,由
"LogAttribute" />生成。- ///
- public class TraceLog
- {
- ///
- /// 当前日志记录的目标类。
- ///
- public string Class { get; internal set; }
- ///
- /// 当前日志跟踪到的异常。
- ///
- public Exception Exception { get; internal set; }
- ///
- /// 当前日志记录的目标方法。
- ///
- public string Method { get; internal set; }
- ///
- /// 当前日志记录的目标方法的参数。
- ///
- public Dictionary
Parameter { get; internal set; }- ///
- /// 当前日志记录的目标方法的返回值。
- ///
- public object ReturnValue { get; internal set; }
- ///
- /// 当前日志记录的目标方法的调用栈。
- ///
- public IEnumerableset; }
- }
关于聊一下Castle DynamicProxy基本用法(AOP)-51CTO.COM到此分享完毕,希望能帮助到您。
本文采摘于网络,不代表本站立场,转载联系作者并注明出处:https://www.iotsj.com//kuaixun/7466.html
用户评论
想了解一下动态代理这种技术,这个文章感觉很有帮助。
有8位网友表示赞同!
看标题就觉得是讲一个很实用的框架,好想去看看怎么使用。
有18位网友表示赞同!
Castle DynamicProxy 这个名字听起来挺熟悉的,之前做过类似的项目好像用过?
有8位网友表示赞同!
AOP 的原理一直都比较模糊,希望这篇文章能解释得更清楚点。
有8位网友表示赞同!
51CTO 上的文章质量比较高,期待这篇介绍的详细内容。
有7位网友表示赞同!
学习一下Castle DynamicProxy 的用法,感觉可以提高开发效率。
有9位网友表示赞同!
最近在想学习动态代理技术,这篇文章正好适合我。
有19位网友表示赞同!
AOP 可以帮助我们做一些代码分离的工作,提升代码的可维护性吧?
有9位网友表示赞同!
文章提到基本用法,是不是入门用得着呢?
有12位网友表示赞同!
看看 Castle DynamicProxy 到底是怎么实现动态代理的。
有10位网友表示赞同!
想了解一下这种技术在实际项目中的应用场景。
有17位网友表示赞同!
感觉动态代理能解决很多重复代码的问题,提高开发效率。
有13位网友表示赞同!
学习 AOP 的目的是为了写出更优雅、更简洁的代码吧?
有6位网友表示赞同!
这个城堡玩意儿听着就高端大气上档次,我倒是想要了解一下。
有14位网友表示赞同!
希望这篇文章能介绍一些经典的案例,方便理解。
有16位网友表示赞同!
动态代理这种技术听起来有点高深啊,希望能简单易懂地讲解。
有5位网友表示赞同!
Castle 这个框架看起来很厉害的样子,希望能够深入了解一下它的用法。
有7位网友表示赞同!
看标题就感到很高兴,终于有人科普 DynamicProxy 了!
有14位网友表示赞同!
AOP 这个概念应该会经常见到吧?还是想要好好学习一番。
有11位网友表示赞同!