IoC和AOP生而就是为了解耦和扩展。
什么是 IoC?一种设计思想,将设计好的对象交给Spring容器控制,而非直接在对象内部控制。
为啥要让容器来管理对象呢?你这程序员咋就知道甩锅呢?
拥有初级趣味的码农,可能只是觉着使用IoC方便,就是个用来解耦的,但这还远非容器的益处。
利用容器管理所有的框架、业务对象,我们可以做到:
- 无侵入调整对象的关系
- 无侵入地随时调整对象的属性
- 实现对象的替换
这使得框架开发者在后续实现一些扩展就很容易。
什么是AOP?AOP实现了高内聚、低耦合,在切面集中实现横切关注点(缓存、权限、日志等),然后通过切点配置把代码注入到合适的位置。
基本概念- 连接点(Join point)
就是方法执行 - 切点(Pointcut)
Spring AOP默认使用AspectJ查询表达式,通过在连接点运行查询表达式来匹配切点 - 增强(Advice)
也叫作通知,定义了切入切点后增强的方式,包括前、后、环绕等。Spring AOP中,把增强定义为拦截器 - 切面(Aspect)
切面=切点+增强
Declaring Advice
有如下三类增强:
@Before
@After
@Around
环绕通知运行 around 匹配方法的执行。它有机会在方法运行之前和之后都工作,并确定方法实际运行何时、如何甚至是否执行。若你需要以线程安全的方式(例如启动和停止计时器)在方法执行前后共享状态,则通常会使用Around advice。
使用案例:
import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.ProceedingJoinPoint; @Aspect public class AroundExample { @Around("com.xyz.myapp.CommonPointcuts.businessService()") public Object doBasicProfiling(ProceedingJoinPoint pjp) throws Throwable { // start stopwatch Object retVal = pjp.proceed(); // stop stopwatch return retVal; } }
around advice 返回的值就是方法调用者看到的返回值。例如,一个简单的缓存aspect可以返回一个值从缓存(如果它有)或调用procedd如果它没有。请注意,可以多次调用procedd,或者根本不在around advice的主体内调用,这都是合法的。
推荐始终使用最不强大的advice形式,以满足需求。
使用 @Around 注解声明环绕通知时,第一个参数必须是ProceedingJoinPoint类型。
在通知的方法体中,调用 proceed() 会导致基础方法运行。 proceed() 也可以在Object[]中传递。数组中的值在进行时用作方法执行的参数。
Advice参数
Spring 提供全种类的通知,这意味着你在通知的方法签名中声明所需参数,而非和Object[]协作。
如何编写通用的通知,以便了解通知方法当前在通知啥玩意。
Access to the Current JoinPoint
任何通知方法都可能声明类型org.aspectj.lang.JoinPoint 的参数(请注意,围绕建议需要申报类型’继续JoinPoint’的第一参数,该参数是 JoinPoint 的子类。
JoinPoint 接口提供了许多有用的方法:
- getArgs()
返回方法的参数 - getThis()
返回代理对象 - getTarget()
返回目标对象 - getSignature()
Returns a description of the method that is being advised. - toString()
Prints a useful description of the method being advised.