一、AOP概念
1、什么是AOP
AOP(Aspect Orient Programming):面向切面编程,可通过运行期,动态代理实现程序功能的统一维护的一种技术。AOP是Spring框架中的一个重要内容。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发效率。
2、AOP的作用及优点
1)让切面功能复用,减少程序中重复代码,
2)让开发人员专注业务逻辑,避免源代码大批量改动
3)实现业务功能和其他非业务功能解耦合,易于维护
4)给存在的业务方法,增加功能,不用修改原来的代码
二、AOP实现过程
AOP代理:可以实现对目标方法的调用,增减功能,保持目标方法内容不变
例如:你自己需要卖账号,自己卖不出去,就需要将账号交给 游戏中介 ,由游戏中介进行代理,类似中介 |
三、AOP编程术语
AOP中重要的三个要素:在Advice的时间,Pointcut位置,执行Aspect
1、切面(Aspect)
切面泛指交叉的业务逻辑,就是给业务方法增加的功能,例如上图的网站曝光。实际就是对业务逻辑的一种增强 |
2、连接点(JoinPoint)
连接切面的业务方法,就是要加强的业务方法,是指可以被切面织入的具体方法。在这个业务方法执行时,会同时执行切面的功能 |
3、切入点(Pointcut)
是一个或多个连接点集合。表示这些方法执行时,都能增加切面的功能,例如要对方法名为update*()的方法加事务。被标记为final的方法是不能作为连接点与切入点的。因为最终的是不能被修改的,不能被增强的 |
4、目标对象(taeget)
给那个对象增加切面的功能,这个对象就是目标对象。即包含主业务逻辑的类的对象。 |
5、通知(Advice)
表示切面的执行时间,在目标之前执行切面,还是目标方法之后执行切面。切入点定义切入的位置,通知定义切入的时间 |
四、AOP技术实现
使用框架实现AOP,实现AOP的框架有很多。有名的两个:
1、Spring
Spring框架实现AOP思想中的部分功能。Spring框架实现AOP的操作比较繁琐,笨重 |
2、Aspectj
独立的框架,专门用于AOP;属于Eclipse,Spring 将AspectJ 对于 AOP 的实现引入到了自己的框架中;在 Spring 中使用 AOP 开发时,一般使用 AspectJ 的实现方式;AspectJ 是一个优秀面向切面的框架,它扩展了 Java 语言,提供了强大的切面实现。 |
五、AspectJ语法
1、通知
Aspectj表示切面执行时间,用的通知(Advice),这个通知可以使用注解表示,常用有下面五个注解:@Before:前置通知 @AfterReturning:后置通知 @Around:环绕通知 @AfterThrowing:异常通知 @After:最终通知 |
2、位置(Pointcut)
Pointcut 用来表示切面执行的位置, 使用Aspectj中切入点表达式。
3、切入点表达式
1)切入点表达式语法
execution(访问权限 方法返回值 方法声明(参数) 异常类型) |
2)切入点表达式原型
execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern? ) |
3)切入点表达式解释
modifiers-pattern? 访问权限类型 execution(访问权限类型 返回值类型 方法名(参数类型和参数个数) 抛出异常类型)
注意,表达式中黑色文字表示可省略部分,各部分间用空格分开。在其中可以使用以下符号:
举例:
|
六、AspectJ基于注解的AOP实现
1、@Before:前置通知
1)前置通知-方法无参数
2)前置通知-方法有 JoinPoint 参数
2、@AfterReturning:后置通知
/*定义方法,表示切面的具体功能 * 后置方法定义 * (1)方法是public (2)方法是void * (3)方法名称自定义(4)方法可以有参数,推荐使用Object类型 * @AfterReturning:后置通知 * 属性:value 切入点表达式 * returning 自定义变量,表示目标方法的返回值的,自定义变量名称必须和通知方法的形参名一样 * 位置:在方法的上面 * 后置通知特点:(1)执行时间:在目标方法之后执行。(2)能获取到目标方法的执行结果。 * (3)不会影响目标方法的执行 * 方法的参数: * JoinPoint joinPoint:必须放在第一位 * Object res:表示目标方法的返回值,使用接受doOther的调用结果。 * Object red = doSome() * 注意: * 1、doSome方法返回的是String,Integer,Lang等基本类型 * 在后置通知中,修改返回值,是不会影响目标方法的最后调用结果的 * 2、doSome返回的结果是对象类型,例如Student * 在后置通知方法中,修改这个Student对象的属性值,是不会影响目标方法的最后调用结果的 */ |
3、 @Around:环绕通知
/*定义方法,表示切面的具体功能 * 环绕通知方法的定义 * (1)方法是public (2)方法是必须有返回值,投建使用Object * (3)方法名称自定义(4)方法必须有ProceedingJoinPoint参数 * @Around:环绕通知 * 属性:value 切入点表达式 * 位置:在方法的上面 * 返回值:Object,表示调用目标方法希望得到的执行结果(不一定是目标方法自己的返回值) * 参数:ProceedingJoinPoint,相当于反射中 Method * 作用:执行目标方法的,等于Method.invoke() * 特点: * 1、在目标方法的前和后都能增强功能 * 2、控制目标方法是否执行 * 3、可以修改目标方法的执行返回结果 */ |
4、@AfterThrowing:异常通知
/*定义方法,表示切面的具体功能
* 异常通知方法的定义
* (1)方法是public (2)方法是没有返回值,是void
* (3)方法名称自定义(4)方法有参数Exception
* @AfterThrowing:异常通知
* 属性:value 切入点表达式
* throwing 自定义变量,表示目标方法抛出的异常:变量名必须和通知方法的形参名一样
* 位置:在方法的上面
* 返回值:Object,表示调用目标方法希望得到的执行结果(不一定是目标方法自己的返回值)
* 参数:ProceedingJoinPoint,相当于反射中 Method
* 作用:执行目标方法的,等于Method.invoke()
* 特点:
* 1、在目标方法抛出异常后执行,没有异常不执行
* 2、能获取到目标方法的异常信息
* 3、不是异常处理程序。可以得到发送异常的通知。可以发送邮件短信,通知开发人员
* 可以看着是目标方法的监控程序
* 异常通知的执行
* try{
* dgsTestServiceXml.doSome(new DgsTestEntity());
* }catch (Exception e){
myAfterThrowing(e);
}
*/
|
5、@After:最终通知
/*定义方法,表示切面的具体功能
* 最终通知方法的定义
* (1)方法是public (2)方法是没有返回值,是void
* (3)方法名称自定义(4)方法没有参数
* @After:最终通知
* 属性:value 切入点表达式
* 位置:在方法的上面
* 特点:
* 1、在目标方法之后执行的
* 2、总是会被执行
* 3、可以用来做程序最后的收尾工作,例如清除临时数据,变量,清理内存
* 最终通知,类似finally
*/ |
6、@Pointcut:定义和管理切入点
/** * @Pointcut:定义和管理切入点,不是通知注解 * 属性:value 切入点表达式 * 位置:在一个自定义方法的上面,这个方法看做是且入点表达式的别名 * 其他的通知注解中,可以使用方法名称,就表示使用这个切入点表达式 **/ |