Spring AOP与代理类的执行顺序级别浅析

 更新时间:2023年03月27日 10:35:09   作者:T.Y.Bao  
这篇文章主要介绍了Spring AOP与代理类的执行顺序级别,关于 Spring AOP和Aspectj的关系,两个都实现了切面编程,Spring AOP更多地是为了Spring框架本身服务的,而Aspectj具有更强大、更完善的切面功能

关于 Spring AOP和Aspectj的关系,两个都实现了切面编程,Spring AOP更多地是为了Spring框架本身服务的,而Aspectj具有更强大、更完善的切面功能,我们在写业务时一般使用AspectJ。不过他们的概念、原理都差不多。

Spring AOP说:

the Spring Framework’s AOP functionality is normally used in conjunction with the Spring IoC container. Aspects are configured using normal bean definition syntax (although this allows powerful “autoproxying” capabilities): this is a crucial difference from other AOP implementations. There are some things you cannot do easily or efficiently with Spring AOP, such as advise very fine-grained objects (such as domain objects typically): AspectJ is the best choice in such cases. However, our experience is that Spring AOP provides an excellent solution to most problems in enterprise Java applications that are amenable to AOP

Spring AOP currently supports only method execution join points (advising the execution of methods on Spring beans). Field interception is not implemented, although support for field interception could be added without breaking the core Spring AOP APIs. If you need to advise field access and update join points, consider a language such as AspectJ.

Spring AOP will never strive to compete with AspectJ to provide a comprehensive AOP solution

可参考 Spring AOP vs AspectJ

AspectJ项目中提供了@AspecJ注解,Spring interprets the same annotations @Aspect as AspectJ 5

Spring AOP提供了定义切面的两种方式

  • schema-based approach : XML-based format 通常不用了
  • @AspectJ annotation style:注解形式,起这个名字是因为借鉴了AspectJ项目,其实用的Spring AOP的注解是@Aspect没有J

注解案例:

@Aspect
@Component 
public class NotVeryUsefulAspect {
	@Pointcut("execution(* transfer(..))")// the pointcut expression
	private void anyOldTransfer() {}// the pointcut signature
	@Before("execution(* com.xyz.myapp.dao.*.*(..))")
    public void doAccessCheck() {
        // ...
    }
 	@AfterReturning(
        pointcut="com.xyz.myapp.SystemArchitecture.dataAccessOperation()",
        returning="retVal")
    public void doAccessCheck(Object retVal) {
        // ...
    }
	@AfterThrowing(
        pointcut="com.xyz.myapp.SystemArchitecture.dataAccessOperation()",
        throwing="ex")
    public void doRecoveryActions(DataAccessException ex) {
        // ...
    }
	@After("com.xyz.myapp.SystemArchitecture.dataAccessOperation()")
    public void doReleaseLock() {
        // ...
    }
	@Around("com.xyz.myapp.SystemArchitecture.businessService()")
    public Object doBasicProfiling(ProceedingJoinPoint pjp) throws Throwable {
        // start stopwatch
        Object retVal = pjp.proceed();
        // stop stopwatch
        return retVal;
    }
}

AOP概念

  • Aspect:cut cross multiple classes. Spring AOP通过 schema-based approach 或 @AspectJ annotation style 两种方式定义切面。
  • Join Point:程序执行点,通常代表一个方法的执行。
  • Advice:Aspect中针对不同Join Point执行的操作,包括 around before after 三种advice。很多AOP框架将advice建模成interceptor,在Join Point外层维护一个 chain of interceptors。
  • Pointcut:匹配 Join Points的规则,如@PointCut("pointcut expression"),满足point expression的将进行切面。
  • Target object:被切面的对象。
  • AOP proxy:代理,Spring Framework通常使用JDK 动态代理或CGLIB动态代理。
  • Weaving:linking aspects with other application types or objects to create an advised object. 发生在编译、加载或运行时。

Type of advice

  • before : @Before
  • after returning : @AfterReturning
  • after throwing : @AfterThrowing
  • after (finally) : @After
  • around : @Around

代理顺序

当一个target object有多个Spring AOP代理时,代理类执行的顺序有时很重要,比如分布式锁代理和事务代理的顺序,分布式锁必须包裹住事务。

先下结论:

Spring通过接口或注解来判断代理的顺序,顺序级别越低,代理越靠内层。顺序级别获取步骤如下:

  • 是否实现org.springframework.core.Ordered接口
  • 是否注解了org.springframework.core.annotation.Order注解
  • 是否注解了javax.annotation.Priority注解
  • 如果都没有,默认最低优先级LOWEST_PRECEDENCE = Integer.MAX_VALUE,如@Transanctional注解就是最低级别

优先级定义范围在Integer.MIN_VALUEInteger.MAX_VALUE(参见Orderd接口)。越小,优先级越高,越在外层先执行。

注解方式生成代理类是通过BeanPostProcessor实现的,由AspectJAwareAdvisorAutoProxyCreator完成,该类实现了BeanPostProcessor

注意到该类有一个Comparator类字段。最终优先级由AnnotationAwareOrderComparator来判断:

public class AnnotationAwareOrderComparator extends OrderComparator {
	/**
	 * Shared default instance of {@code AnnotationAwareOrderComparator}.
	 */
	public static final AnnotationAwareOrderComparator INSTANCE = new AnnotationAwareOrderComparator();
		@Override
	@Nullable
	protected Integer findOrder(Object obj) {
		// 父类,判断是否实现了Orderd接口
		// (obj instanceof Ordered ? ((Ordered) obj).getOrder() : null)
		Integer order = super.findOrder(obj);
		if (order != null) {
			return order;
		}
		// 没有实现则继续搜寻@Order和@Priority注解
		// 如果返回null,表示找不到,会返回父类执行getOrder方法返回Ordered.LOWEST_PRECEDENCE
		return findOrderFromAnnotation(obj);
	}
	@Nullable
	private Integer findOrderFromAnnotation(Object obj) {
		...
		// 找@Order和@Priority注解
		Integer order = OrderUtils.getOrderFromAnnotations(element, annotations);
		// 如果该类没有注解,递归找被代理类
		if (order == null && obj instanceof DecoratingProxy) {
			return findOrderFromAnnotation(((DecoratingProxy) obj).getDecoratedClass());
		}
		// 被代理类有则返回优先级,没有则返回null
		return order;
	}
}

到此这篇关于Spring AOP与代理类的执行顺序级别浅析的文章就介绍到这了,更多相关Spring AOP与代理类内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • IDEA新手必备之各种快捷键详解

    IDEA新手必备之各种快捷键详解

    这篇文章主要介绍了IDEA新手必备之各种快捷键详解,文中有非常详细的快捷键介绍,对正在使用IDEA的小伙伴们有非常好的帮助,需要的朋友可以参考下
    2021-04-04
  • Java实现鼠标模拟与键盘映射

    Java实现鼠标模拟与键盘映射

    这篇文章主要为大家详细介绍了Java实现鼠标模拟与键盘映射,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-08-08
  • Mybatis注解方式完成输入参数为list的SQL语句拼接方式

    Mybatis注解方式完成输入参数为list的SQL语句拼接方式

    这篇文章主要介绍了Mybatis注解方式完成输入参数为list的SQL语句拼接方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • IDEA 创建一个Mybatis Maven项目的方法步骤(图文)

    IDEA 创建一个Mybatis Maven项目的方法步骤(图文)

    这篇文章主要介绍了IDEA 创建一个Mybatis Maven项目的方法步骤(图文),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-03-03
  • 详解Java使用Jsch与sftp服务器实现ssh免密登录

    详解Java使用Jsch与sftp服务器实现ssh免密登录

    这篇文章主要介绍了详解Java使用Jsch与sftp服务器实现ssh免密登录,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-10-10
  • Java 读取图片的mimeType的方法

    Java 读取图片的mimeType的方法

    本篇文章主要介绍了Java 读取图片的mimeType的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-01-01
  • Java中的原子类AtomicInteger使用详解

    Java中的原子类AtomicInteger使用详解

    这篇文章主要介绍了Java中的原子类AtomicInteger使用详解,原子操作是指不会被线程调度机制打断的操作,这种操作一旦开始,就一直运行到结束,中间不会有任何线程上下文切换,需要的朋友可以参考下
    2023-12-12
  • 浅谈一下Spring的核心基础IOC与DI

    浅谈一下Spring的核心基础IOC与DI

    这篇文章主要介绍了Spring的核心基础IOC与DI的详细用法,spring技术是现在企业开发中几乎必备的技术选型,那么学好spring就很重要,本篇着重讲解spring的核心机制,IOD与DI,一起来看看吧
    2023-03-03
  • java利用POI读取excel文件的方法

    java利用POI读取excel文件的方法

    这篇文章主要介绍了java利用POI读取excel文件的方法,帮助大家更好的理解和学习Java,感兴趣的朋友可以了解下
    2020-12-12
  • 解决SpringMvc中普通类注入Service为null的问题

    解决SpringMvc中普通类注入Service为null的问题

    这篇文章主要介绍了解决SpringMvc中普通类注入Service为null的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-07-07

最新评论