AOP基本组成部分
1.注解
(1)Aspect(切面):通常是一个类,里面可以定义切入点和通知
(2)JointPoint(连接点):程序执行过程中明确的点,一般是方法的调用
(3)Advice(通知):AOP在特定的切入点上执行的增强处理,有before,after,afterReturning,afterThrowing,around
(4)Pointcut(切入点):就是带有通知的连接点,在程序中主要体现为书写切入点表达式
(5)AOP代理:AOP框架创建的对象,代理就是目标对象的加强。Spring中的AOP代理可以使JDK动态代理,也可以是CGLIB代理,前者基于接口,后者基于子类
同时基于注解的AOP配置方式
1.启用@AsjectJ支持
在applicationContext.xml中配置下面一句:
<aop:aspectj-autoproxy />
2.切入点表达式
切入点指示符用来指示切入点表达式目的,,在Spring AOP中目前只有执行方法这一个连接点,Spring AOP支持的AspectJ切入点指示符如下:
execution:用于匹配方法执行的连接点;
within:用于匹配指定类型内的方法执行;
this:用于匹配当前AOP代理对象类型的执行方法;注意是AOP代理对象的类型匹配,这样就可能包括引入接口也类型匹配;
target:用于匹配当前目标对象类型的执行方法;注意是目标对象的类型匹配,这样就不包括引入接口也类型匹配;
args:用于匹配当前执行的方法传入的参数为指定类型的执行方法;
@within:用于匹配所以持有指定注解类型内的方法;
@target:用于匹配当前目标对象类型的执行方法,其中目标对象持有指定的注解;
@args:用于匹配当前执行的方法传入的参数持有指定注解的执行;
@annotation:用于匹配当前执行方法持有指定注解的方法;
实现一个功能:controller层所有方法,入参和出参的自动打印
思路如下:
自定义注解,针对注解创建切面,对Controller 中的方法使用注解,通过切面中的 @Before, @Around 获取数据并打印
先导入pom依赖
<!-- spring-aop -->
<spring.version>4.3.12.RELEASE</spring.version>
<!-- jackson-databind, jackson-core, jackson-annotations -->
<jackson.version>2.9.1</jackson.version>
<!-- aspectjrt, aspectjweaver 实现切面 -->
<aspectj.version>1.8.9</aspectj.version>
<!-- logback-core 会自动引入依赖的 sjf4j -->
<logback.version>1.2.2</logback.version>
<!-- lombok sjf4j 中 log 对象实例化-->
<lombok.version>1.16.18</lombok.version>
spring-mvc.xml 添加 aop 自动代理 <aop:aspectj-autoproxy/>
新建注解 LogPrint,将注解作为切点,作用到 Controller 的方法
@Retention(RetentionPolicy.RUNTIME) @Target(ElementType.METHOD) public @interface LogPrint { String description() default ""; }
新建切面 LogPrintAspect,在 controller 方法调用前,调用后打印入参,出参
、
@Aspect @Component @Slf4j public class LogPrintAspect { @Pointcut("@annotation(cn.exrick.common.annotaion.LogPrint)") public void logPrint(){} joinPoint.getArgs() -- 获取入参 @Before("logPrint()") public void doBefore(JoinPoint joinPoint){ proceedingJoinPoint.proceeding() -- 执行切点修饰的方法,拿到结果,打印响应 @Around("logPrint()") public Object doAround(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {} @After("logPrint()") public void doAfter(){} }
使用注解
@LogPrint
修饰 Controller 类中的方法@RestController public class LogController { @LogPrint @RequestMapping("/logGet") public Object getTest(String name, String age){ String content = name + "今年" + age + "岁"; HashMap<String, Object> resultMap = new HashMap<>(); resultMap.put("content", content); return resultMap; } }