Spring5学习笔记2AOP
Spring5学习笔记2——AOP
配套视频参考:孙哥说Spring5
第二部分:AOP编程
第一章、静态代理设计模式
1. 为什么需要代理设计模式
-
在JavaEE分层开发开发中,那个层次对于我们来讲最重要
1
2
3DAO ---> Service --> Controller
JavaEE分层开发中,最为重要的是Service层 -
Service层中包含了哪些代码?
1
2
3
4
5
6
7
8Service层中 = 核心功能(几十行 上百代码) + 额外功能(附加功能)
1. 核心功能
业务运算
DAO调用
2. 额外功能(事务、日志、性能...)
1. 不属于业务
2. 可有可无
3. 代码量很小 -
额外功能书写在Service层中好不好?
Controller层(Service层的调用者)除了需要核心功能,还需要这些额外功能。
但是从软件设计者角度看:Service层最好不要写额外功能。
-
现实生活中的解决方式
2. 代理设计模式分析
2.1 概念
通过代理类,为原始类(目标)增加额外的功能
好处:利于原始类(目标)的维护
2.2 名词解释
1 | 1. 目标类 原始类 |
2.3 代理开发的核心要素
1 | 代理类 = 实现和目标类相同的接口 + 在同名方法中添加额外功能 + 调用原始类同名方法 |
2.4 编码
静态代理:为每一个原始类,手动编写一个代理类 (.java .class)
2.5 静态代理存在的问题
1 | 1. 静态类文件数量过多,不利于项目管理 |
第二章、Spring的动态代理开发
1. Spring动态代理的概念
1 | 概念:通过代理类为原始类(目标类)增加额外功能,代理类由Spring动态生成。 |
2. 搭建开发环境
1 | <dependency> |
3. Spring动态代理的开发步骤
3.1 创建原始对象(目标对象)
1 | public class UserServiceImpl implements UserService{ |
3.2 定义额外功能(实现MethodBeforeAdvice接口)
1 | public class Before implements MethodBeforeAdvice { |
3.3 定义切入点
1 | 1. 切入点:额外功能加入的位置 |
3.4 组装
1 | <!-- 组装切入点与额外功能 --> |
3.5 调用
1 | 目的:获得Spring工厂创建的动态代理对象,并进行调用 |
4. 动态代理细节分析
4.1 Spring创建的动态代理类在哪里?
1 | Spring框架在运行时,通过动态字节码技术,在JVM创建的,运行在JVM内部,等程序结束后就消失了。 |
4.2 动态代理编程简化代理的开发
在额外功能不改变的前提下,创建其他目标类(原始类)的代理对象时,只需要指定原始(目标)对象即可。
4.3 动态代理额外功能的维护性大大增强
第三章、Spring动态代理详解
1. 额外功能的详解
-
MethodBeforeAdvice分析
作用:原始方法执行之前,运行额外功能。
1
2
3
4
5
6
7
8
9
10
11
12
13
14public class Before implements MethodBeforeAdvice {
/**
* 作用:给原始方法添加额外功能
* 注意:会在原始方法运行之前运行此方法
* @param method 原始方法 login() register() ...
* @param objects 原始方法的参数列表 name password ...
* @param o 原始对象 UserServiceImpl OrderServiceImpl
* @throws Throwable 抛出的异常
*/
public void before(Method method, Object[] objects, Object o) throws Throwable {
System.out.println("---- MethodBeforeAdvice log... ----");
}
}
实战:需要时才用,可能都会用到,也有可能都不用。
-
MethodInterceptor(方法拦截器)
MethodInterceptor
接口:额外功能可定义在原始方法执行 前、后、前和后。1
2
3
4
5
6
7
8
9
10
11
12
13
14
15public class Around implements MethodInterceptor {
/**
* @param invocation 封装了原始方法 invocation.proceed()表示原始方法的运行
* @return 原始方法的返回值
* @throws Throwable 可能抛出的异常
*/
public Object invoke(MethodInvocation invocation) throws Throwable {
System.out.println("------ 额外功能 log -----");
//原始方法的执行
Object ret = invocation.proceed();
//返回原始方法的返回值
return ret;
}
}
额外功能运行在原始方法执行之后
1 | public Object invoke(MethodInvocation invocation) throws Throwable { |
额外功能运行在原始方法执行之前和之后(实战:事务需要在之前和之后都运行)
1 |
|
额外功能运行在原始方法抛出异常时
1 |
|
MethodInterceptor可以影响原始方法的返回值
1 |
|
2. 切入点详解
切入点决定了额外功能加入的位置。
1 | <aop:pointcut id="pc" expression="execution(* *(..))"/> |
2.1 切入点表达式
-
方法切入点表达式:
1
2
3
4
5
6* *(..) --> 所有方法
* ---> 修饰符 返回值
* ---> 方法名
()---> 参数表
..---> 对于参数没有要求 (0个或多个)
举例:
1 | # 定义login方法作为切入点 |
-
类切入点表达式:
指定特定的类作为切入点,即这个类中所有的方法都会加上额外功能。
举例:
1
2
3
4
5
6
7
8
9# 类中的所有方法都加入额外功能
* com.yuziyan.proxy.UserServiceImpl.*(..)
# 忽略包
# 1. 类只在一级包下 com.UserServiceImpl
* *.UserServiceImpl.*(..)
# 2. 类可在多级包下 com.yuziyan.proxy.UserServiceImpl
* *..UserServiceImpl.*(..) -
包切入点表达式:
指定包作为切入点,即这个包中的所有类及其方法都会加入额外的功能。
举例:
1
2
3
4
5# proxy包作为切入点,即proxy包下所有类中的所有方法都会加入额外功能,但是不包括其子包中的类!
* com.yuziyan.proxy.*.*(..)
# 当前包及其子包都生效
* com.yuziyan.proxy..*.*(..)
2.5 切入点函数
作用:用于执行切入点表达式。
-
execution()
1
2
3
4
5
6
7最为重要的切入点函数,功能最全!
用于执行:方法切入点表达式、类切入点表达式、包切入点表达式
弊端:execution执行切入点表达式 ,书写麻烦
execution(* com.yuziyan.proxy..*.*(..))
注意:其他的切入点函数 只是简化execution书写复杂度,功能上完全一致 -
args()
1
2
3
4
5
6# 作用:用于函数(方法)参数的匹配
# 举例:方法参数必须得是2个字符串类型的参数
execution(* *(String,String))
等价于:
args(String,String) -
within()
1
2
3
4
5
6
7
8
9
10# 作用:用于进行类、包切入点表达式的匹配
# 举例:
# UserServiceImpl类作为切入点:
execution(* *..UserServiceImpl.*(..))
within(*..UserServiceImpl)
# proxy包作为切入点:
execution(* com.yuziyan.proxy..*.*(..))
within(com.yuziayan.proxy..*) -
@annotation()
1
2
3<!-- 作用:为具有特殊注解的方法加入额外功能 -->
<aop:pointcut id="" expression="@annotation(com.baizhiedu.Log)"/> -
切入点函数间的逻辑运算:
目的:整合多个切入点函数一起配合工作,进而完成更为复杂的需求。
-
and 与操作(同时满足)
1
2
3
4
5
6
7
8
9# 案例:方法名为login,同时有2个字符串类型的参数:
execution(* login(String,String))
execution(* login(..)) and args(String,String)
# 注意:与操作不能用于同种类型的切入点函数
# 错误案例:register方法 和 login方法作为切入点(不能用and,而用or!)
execution(* login(..)) and execution(* register(..))
# 上面的语句会发生错误,因为其实际表达的含义是方法名为login同时方法名为register,显然有悖逻辑,此时应该用到的是 or
-
-
or 或操作(满足一种即可)
1
2# 案例:register方法 和 login方法作为切入点
execution(* login(..)) or execution(* register(..))
第四章、AOP编程
1. AOP概念
1 | AOP (Aspect Oriented Programing) 面向切面编程 = Spring动态代理开发 |
2. AOP编程的开发步骤
- 原始对象
- 额外功能 (MethodInterceptor)
- 切入点
- 组装切面 (额外功能+切入点)
3. 切面的名词解释
1 | 切面 = 切入点 + 额外功能 |
第五章、AOP的底层实现原理
1. 核心问题
- AOP如何创建动态代理类?(动态字节码技术)
- Spring工厂如何加工创建代理对象?通过原始对象的id值,获得的是代理对象。
2. 动态代理类的创建
2.1 JDK的动态代理
-
Proxy.newProxyInstance()
方法参数详解:
-
编码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25public class TestJDKProxy {
public static void main(String[] args) {
//1.创建原始对象
//注意:由于后面匿名子类的方法中用到了userService,所以应该用final修饰
// 而JDK1.8以后默认加了final,不需要手动加
UserService userService = new UserServiceImpl();
//2.JDK创建代理对象
InvocationHandler handler = new InvocationHandler() {
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("----------- JDKProxy log -----------");\
//目标方法运行:
Object ret = method.invoke(userService, args);
return ret;
}
};
UserService userServiceProxy = (UserService) Proxy.newProxyInstance(userService.getClass().getClassLoader(), userService.getClass().getInterfaces(),handler);
userServiceProxy.login("海绵宝宝", "1111");
userServiceProxy.register(new User());
}
}
2.2 CGlib的动态代理
- 原理:通过父子继承关系创建代理对象。原始类作为父类,代理类作为子类,这样既可以保证2者方法一致,同时在代理类中提供新的实现(额外功能+原始方法)
-
CGlib编码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41package com.yuziyan.cglib;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
public class TestCGlibProxy {
public static void main(String[] args) {
//1.创建原始对象
UserServiceImpl userService = new UserServiceImpl();
//2.通过CGlib创建代理对象
// 2.1 创建Enhancer
Enhancer enhancer = new Enhancer();
// 2.2 设置借用类加载器
enhancer.setClassLoader(TestCGlibProxy.class.getClassLoader());
// 2.3 设置父类(目标类)
enhancer.setSuperclass(userService.getClass());
// 2.4 设置回调,额外功能写在里面
enhancer.setCallback(new MethodInterceptor() {
//相当于 InvocationHandler --> invoke()
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
//额外功能:
System.out.println("========= CGlibProxy log ========");
//目标方法执行:
Object ret = method.invoke(userService, objects);
return ret;
}
});
// 2.5 通过Enhancer对象创建代理
UserServiceImpl service = (UserServiceImpl) enhancer.create();
//测试:
service.register();
service.login();
}
}
2.3 总结
1 | 1. JDK动态代理 Proxy.newProxyInstance() 通过目标类实现的接口创建代理类 |
3. Spring工厂如何返回代理对象
- 思路分析:
-
编码模拟:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22public class ProxyBeanPostProcessor implements BeanPostProcessor {
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
return bean;
}
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
InvocationHandler invocation = new InvocationHandler() {
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("----------- 模拟Spring返回代理对象的方式 log -----------");
Object ret = method.invoke(bean, args);
return ret;
}
};
return Proxy.newProxyInstance(ProxyBeanPostProcessor.class.getClassLoader(), bean.getClass().getInterfaces(), invocation);
}
}
1 | <!-- 1.配置原始对象 --> |
第六章、基于注解的AOP编程
1. 开发步骤:
-
原始对象
-
额外功能
-
切入点
-
组装切面
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19/**
* 声明切面类 @Aspect
* 定义额外功能 @Around
* 定义切入点 @Around("execution(* login(..))")
*
*/
public class MyAspect {
//组装了切入点和额外功能
public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
//额外功能:
System.out.println("--------- 基于注解的AOP编程 log --------");
//原始方法执行:
Object ret = joinPoint.proceed();
return ret;
}
}
1 | <!-- 原始对象 --> |
2. 细节分析:
-
切入点复用:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
public class MyAspect {
/**
* 切入点复用:定义一个函数,加上@Pointcut注解,通过该注解的value定义切入点表达式,以后可以复用。
*/
public void myPointcut(){}
//组装了切入点和额外功能
public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
//额外功能:
System.out.println("--------- 基于注解的AOP编程 log --------");
//原始方法执行:
Object ret = joinPoint.proceed();
return ret;
}
public Object around1(ProceedingJoinPoint joinPoint) throws Throwable {
//额外功能:
System.out.println("--------- 基于注解的AOP编程 tx --------");
//原始方法执行:
Object ret = joinPoint.proceed();
return ret;
}
} -
动态代理的创建方式:
1
2
3
4
5
6
7
8
9
10
11AOP底层实现 2种代理创建方式
1. JDK 通过实现接口,创建代理对象
2. Cglib 通过继承目标类,创建代理对象
默认情况 AOP编程 底层应用JDK动态代理创建方式
如果要切换Cglib
1. 基于注解AOP开发
<aop:aspectj-autoproxy proxy-target-class="true" />
2. 传统的AOP开发
<aop:config proxy-target-class="true">
</aop>
第七章、AOP开发中的一个坑
坑:在同一个业务类中,业务方法间相互调用时,只有最外层的方法,加入了额外功能(内部的方法,通过普通的方式调用,运行的都是原始方法)。如果想让内层的方法也调用代理对象的方法,就要实现AppicationContextAware获得工厂,进而获得代理对象。
1 | public class UserServiceImpl implements UserService, ApplicationContextAware { |