文章目录
一、什么是AOP?
1、面向切面编程(方面),利用 AOP 可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
2、通俗描述:不通过修改源代码方式,在主干功能里面添加新功能
Aop在Spring中的作用
提供声明式事务;允许用户自定义切面
AOP中的专业术语
- 连接点:类里面哪些方法可以被增强,这些方法称为连接点
- 切入点:实际被真正增强的方法,称为切入点
- 通知(增强):实际增强的逻辑部分称为通知(增强)
- 切面:把通知应用到切入点的过程
- 注意点:通知有多种类型,如前置通知、后置通知、环绕通知、异常通知、最终通知
二、实现方式(jdk动态代理)
1、方式一(通过 Spring API 实现)
前提:需要导入AOP依赖包:
<!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver -->
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.4</version>
</dependency>
第一步:编写我们的业务接口和实现类
public interface UserService {
public void add();
public void delete();
public void update();
public void search();
}
public class UserServiceImpl implements UserService{
@Override
public void add() {
System.out.println("增加用户");
}
@Override
public void delete() {
System.out.println("删除用户");
}
@Override
public void update() {
System.out.println("更新用户");
}
@Override
public void search() {
System.out.println("查询用户");
}
}
第二步:然后去写我们的增强类 , 我们编写两个 , 一个前置增强 一个后置增强
public class Log implements MethodBeforeAdvice {
//method : 要执行的目标对象的方法
//objects : 被调用的方法的参数
//Object : 目标对象
@Override
public void before(Method method, Object[] objects, Object o) throws Throwable {
System.out.println( o.getClass().getName() + "的" + method.getName() + "方法被执行了");
}
}
public class AfterLog implements AfterReturningAdvice {
//returnValue 返回值
//method被调用的方法
//args 被调用的方法的对象的参数
//target 被调用的目标对象
@Override
public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
System.out.println("执行了" + target.getClass().getName()
+"的"+method.getName()+"方法,"
+"返回值:"+returnValue);
}
}
第三步:最后去spring的文件中注册 , 并实现aop切入实现 , 注意导入约束 .
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<!--注册bean-->
<bean id="userService" class="com.kuang.service.UserServiceImpl"/>
<bean id="log" class="com.kuang.log.Log"/>
<bean id="afterLog" class="com.kuang.log.AfterLog"/>
<!--aop的配置-->
<aop:config>
<!--切入点 expression:表达式匹配要执行的方法-->
<aop:pointcut id="pointcut" expression="execution(* com.kuang.service.UserServiceImpl.*(..))"/>
<!--执行环绕; advice-ref执行方法 . pointcut-ref切入点-->
<aop:advisor advice-ref="log" pointcut-ref="pointcut"/>
<aop:advisor advice-ref="afterLog" pointcut-ref="pointcut"/>
</aop:config>
</beans>
其中execution(* com.kuang.service.UserServiceImpl.*(…))
解释如下:
符号 | 含义 |
---|---|
execution() | 表达式的主体; |
第一个”*“符号 | 表示返回值的类型任意; |
com.kuang.service.UserServiceImpl | AOP所切的服务的包名,即,我们的业务部分 |
包名后面的”.“ | 表示当前包及子包 |
.*(…) | 表示任何方法名,括号表示参数,两个点表示任何参数类型 |
第四步:测试
public class MyTest {
@Test
public void test(){
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
//注意这里调用的是接口而不是实现类
UserService userService = context.getBean("userService",UserService.class);
userService.search();
}
}
2、方式二(自定义类实现AOP)
自定义类来实现Aop
目标业务类不变依旧是userServiceImpl
第一步 : 自定义一个切入类
public class DiyPointcut {
public void before(){
System.out.println("---------方法执行前---------");
}
public void after(){
System.out.println("---------方法执行后---------");
}
}
第二步:去spring中配置
<!--第二种方式自定义实现-->
<!--注册bean-->
<!--切入的目标类-->
<bean id="userService" class="com.zhao.service.UserServiceImpl"></bean>
<!--自定义的切入类即切面-->
<bean id="diyCut" class="com.zhao.diy.DiyPointCut"></bean>
<!--aop的配置-->
<aop:config>
<!--第二种方式:使用AOP的标签实现-->
<aop:aspect ref="diyCut">
<aop:pointcut id="diyPonitcut" expression="execution(* com.kuang.service.UserServiceImpl.*(..))"/>
<aop:before pointcut-ref="diyPonitcut" method="before"/>
<aop:after pointcut-ref="diyPonitcut" method="after"/>
</aop:aspect>
</aop:config>
第三步:测试
public class MyTest {
@Test
public void test(){
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
UserService userService = context.getBean("userService",UserService.class);
userService.add();
}
}
3、方式三(基于注解的方式进行实现)
第一步:编写一个注解实现的增强类
package com.zhao.diy;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
@Aspect
public class AnnotationPointCut {
@Before("execution(* com.zhao.service.UserServiceImpl.*(..))")
public void before(){
System.out.println("---------方法执行前---------");
}
@After("execution(* com.zhao.service.UserServiceImpl.*(..))")
public void after(){
System.out.println("---------方法执行后---------");
}
//相同切入点抽取
@Pointcut(value = "execution(* com.zhao.service.UserServiceImpl.*(..))")
public void pointDemo() {
}
@Around(value = "pointDemo()")
public void around(ProceedingJoinPoint jp) throws Throwable {
System.out.println("环绕前");
System.out.println("签名:"+jp.getSignature());
//执行目标方法proceed
Object proceed = jp.proceed();
System.out.println("环绕后");
}
}
此外,在使用注解时,为了增强减少代码冗余,可以创建一个方法来替代execution表达式。如:
//相同切入点抽取
@Pointcut(value = "execution(* com.zhao.service.UserServiceImpl.*(..))")
public void pointDemo() {
}
@Around(value = "pointDemo()")
public void around(ProceedingJoinPoint jp) throws Throwable {
System.out.println("环绕前");
System.out.println("签名:"+jp.getSignature());
//执行目标方法proceed
Object proceed = jp.proceed();
System.out.println("环绕后");
}
第二步:在Spring配置文件中,注册bean,并增加支持注解的配置
<!--注册bean-->
<bean id="userServiceImpl" class="com.zhao.service.UserServiceImpl"></bean>
<!--在Spring配置文件中,注册bean,并增加支持注解的配置-->
<bean id="pointCut" class="com.zhao.diy.AnnotationPointCut"></bean>
<!--开启自动注解-->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>
总结
Spring的Aop就是将公共的业务 (日志 , 安全等) 和领域业务结合起来 , 当执行领域业务时 , 将会把公共业务加进来 . 实现公共业务的重复利用 . 领域业务更纯粹 , 程序猿专注领域业务 , 其本质还是动态代理 。