目录
什么是AOP
AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期间动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
AOP相关名词
横切关注点:跨越应用程序多个模块的方法或功能。即是,与我们业务逻辑无关的,但是我们需要关注的部分,就是横切关注点。如日志 , 安全 , 缓存 , 事务等等 ....
切面(ASPECT):横切关注点 被模块化 的特殊对象。即,它是一个类。
通知(Advice):切面必须要完成的工作。即,它是类中的一个方法。
目标(Target):被通知对象(接口或方法)。
代理(Proxy):向目标对象应用通知之后创建的对象。
切入点(PointCut):切面通知执行的 “地点”的定义。
连接点(JointPoint):与切入点匹配的执行点。
SpringAOP中,通过Advice定义横切逻辑,Spring中支持5种类型的Advice:
使用Spring实现AOP
准备:导入依赖包
<!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver -->
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.9.1</version>
<scope>runtime</scope>
</dependency>
方式一:使用Spring的API接口
- 实现MethodBeforeAdvice 和 AfterReturningAdvice方法,确定要放在切入点前或者后
- 配置<aop:pointcut 确定切入点位置
- 配置<aop:advisor 把类和切入点关联,这样就能通过类里面的方法和切入点位置实现AOP
- 注意动态代理的是接口,正常来说使用方法必须使用实现类,但AOP代理的是接口,只需要使用接口即可
Log.java
public class Log implements MethodBeforeAdvice {
//method:要执行的目标对象的方法
//args:参数
//target:目标对象
public void before(Method method, Object[] args, Object target) throws Throwable {
System.out.println(target.getClass().getName()+"的"+method.getName()+"被执行了");
}
}
AfterLog.java
public class AfterLog implements AfterReturningAdvice {
//returnValue:返回值
public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
System.out.println("执行了"+method.getName()+"返回结果为:"+returnValue);
}
}
applicationContext.xml
<!--注册bean-->
<bean id="userService" class="com.lys.service.UserServiceImpl"/>
<bean id="log" class="com.lys.log.Log"/>
<bean id="afterLog" class="com.lys.log.AfterLog"/>
<!--方式一:使用原生Spring API接口-->
<!--配置aop:需要导入aop约束-->
<aop:config>
<!--切入点-->
<!--execution表达式
execution(修饰词(可省略)、返回值、全类名、方法名、参数)
..表示任意参数,*表示所有类
定位切入点位置-->
<aop:pointcut id="pointcut" expression="execution(* com.lys.service.UserServiceImpl.*(..))"/>
<!--执行环绕增加-->
<aop:advisor advice-ref="log" pointcut-ref="pointcut"/>
<aop:advisor advice-ref="afterLog" pointcut-ref="pointcut"/>
</aop:config>
MyTest.java
public class MyTest {
public static void main(String[] args) {
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
//动态代理代理的是接口
UserService userService = context.getBean("userService", UserService.class);
userService.add();
}
}
方式二:使用自定义类实现AOP
<aop:config>
<!--自定义切面,ref 要引用的类-->
<aop:aspect ref="diy">
<!--切入点-->
<aop:pointcut id="point" expression="execution(* com.lys.service.UserServiceImpl.*(..))"/>
<!--通知-->
<aop:before method="before" pointcut-ref="point"/>
<aop:after method="after" pointcut-ref="point"/>
</aop:aspect>
</aop:config>
方式三:注解实现AOP
<!--开启注解支持-->
<aop:aspectj-autoproxy/>
@Aspect //标注这个类是一个切面
public class Annotation {
//第一种插入: before / after插入
@Before("execution(* com.lys.service.UserServiceImpl.*(..))")
public void before(){
System.out.println("===========方法执行前===========");
}
@After("execution(* com.lys.service.UserServiceImpl.*(..))")
public void after(){
System.out.println("===========方法执行后===========");
}
//第二组插入:在环绕增强插入
@Around("execution(* com.lys.service.UserServiceImpl.*(..))")
public void around(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("环绕前");
Signature signature = joinPoint.getSignature();//获得签名
System.out.println("signature:"+signature);
//执行方法
Object proceed = joinPoint.proceed();
System.out.println("环绕后");
}
}