关于Spring AOP
一.AOP:
1.什么是AOP:
在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。
AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
AOP采取横向抽取机制,取代了传统纵向继承体系重复性代码。
经典应用:性能监视、事务管理、安全检查、缓存等。
Spring AOP使用纯Java实现,不需要专门的编译过程和类加载器,在运行期通过代理方式向目标类织入增强代码。
AspectJ是一个基于Java语言的AOP框架,Spring2.0开始,Spring AOP引入对Aspect的支持,AspectJ扩展了Java语言,提供了一个专门的编译器,在编译时提供横向代码的织入。
2.Aop实现原理:
Aop 底层采用是代理机制。
接口 + 实现类:有接口,采用JDK动态代理。
实现类:只有实现类,没有接口,采用CGLIB 字节码增强。
3.Aop相关术语
Target : 目标类,需要被增强的类。
JoinPoint:连接点,目标类上需要被增强的方法。(这些方法可以被增强,也可能不增强)。
PointCut:切入点,被增强的连接点。(已经增强了)。
切入点可以理解为是连接点一个子集。
Advice :增强/通知,增强的方法。
weaving:织入,将切入点和通知结合,生成代理类过程。
Proxy:代理类。
Aspect:切面,切入点和通知结合(切入点和 通知点 多点形成面)。
特殊情况:一个切入点和 一个通知。
Introduction(引介):特殊的通知,可以对类增强,添加方法或字段。
代理知识总结:
1.Spring在运行期,生成动态代理对象,不需要特殊的编译器;
2.Spring AOP的底层就是通过JDK动态代理或CGLib动态代理技术 为目标Bean执行横向织入;
3.若目标对象实现了若干接口,spring使用JDK的java.lang.reflect.Proxy类代理;
4.若目标对象没有实现任何接口,spring使用CGLIB库生成目标对象的子类;
5.程序中应优先对接口创建代理,便于程序解耦维护;
6.标记为final的方法,不能被代理,因为无法进行覆盖;
7.JDK动态代理,是针对接口生成子类,接口中方法不能使用final修饰;
8.CGLib 是针对目标类生产子类,因此类或方法 不能使final的;
9.Spring只支持方法连接点,不提供属性连接;
二.Spring工厂bean
1.Aop联盟通知类型
使用spring提供 FactoryBean创建代理对象,手动的获取代理对象。
生成代理需要应用增强(通知),通知需要确定方法名称。spring规范规定通知类型,从而确定方法名称。
Aop联盟确定5中通知类型,Spring对Aop联盟进行支持。
前置通知org.springframework.aop.MethodBeforeAdvice
在目标方法执行前实施增强
后置通知org.springframework.aop.AfterReturningAdvice
在目标方法执行后实施增强
环绕通知org.aopalliance.intercept.MethodInterceptor【】
在目标方法执行前后实施增强
异常抛出通知org.springframework.aop.ThrowsAdvice
在方法抛出异常后实施增强
引介通知org.springframework.aop.IntroductionInterceptor(了解)
在目标类中添加一些新的方法和属性
环绕
try{
前置
//必须手动执行目标方法
后置
} catch(){
//异常抛出
}
三.AspectJaop框架:
1.AspectJ是一个基于Java语言的AOP框架。
Spring2.0以后新增了对AspectJ切入点表达式支持。
@Aspect是AspectJ1.5新增功能,通过JDK5注解技术,允许直接在Bean类中定义切面。
新版本Spring框架,建议使用AspectJ方式来开发AOP
导入jar包:
aop联盟:com.springsource.org.aopalliance-1.0.0.jar
springaop支持:spring-aop-3.2.0.RELEASE.jar
aspect规范:com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar
springaspect 支持:spring-aspects-3.2.0.RELEASE.jar
2.切入点表达式
1.execution:匹配方法的执行
格式:execution(修饰符返回值类型 包.类名.方法(参数) throws 异常)
1.1修饰符,表示方法的修饰符,一般省略。
1.2返回值类型
String 表示返回String
void 表示没有返回值
* 表示返回任意类型【】
1.3包
1.4类名
UserService 表示指定的类
*Service 表示以Service结尾
Test* 表示以Test开头
* 表示任意类名
1.5方法名
addUser 表示指定方法
add* 表示以add开头
*Do 表示以Do结尾
* 表示任意
1.6参数列表
() 表示没有参数
(int) 表示一个参数int类型
(int,int) 表示两个参数int类型
(如果是java.lang包下的可以省略,其他都必须是全限定类名)
(..) 表示参数任意
1.7throws 异常, 一般省略。
2.within:匹配包或子包中的方法
within(cn.itcast.aop..*)
3.this:匹配实现接口的代理对象中的方法
this(cn.itcast.aop.user.UserDAO)
4.target:匹配实现接口的目标对象中的方法
target(cn.itcast.aop.user.UserDAO)
5.args:匹配参数格式符合标准的方法
args(int,int)
6.bean(名称) ,匹配指定的bean
bean("userServiceId")
基于注解 aspectJ:
步骤一:将切面类配置给spring
@Component
public class MyAspect {}
步骤二:将切面类声明成切面
@Component
@Aspect
public class MyAspect {}
步骤三:声明共有切入点
1.方法必须private,没有返回值,没有参数
2.之后使用将其当成方法调用。例如:@After("myPointcut()")
@Pointcut("execution(*com.itheima.d_aspect.b_annotation.*.*(..))")
privatevoid myPointcut(){}
步骤四:编写相应的通知
@Before 前置
@AfterReturning 后置,可以获得返回值,必须在注解中确定返回值参数名称。
@AfterThrowing 抛出异常,可以获得具体异常信息,必须在注解确定第二个参数名称
@Around 环绕
@After 最终