Spring-AOP @AspectJ进阶之绑定连接点方法入参

简介: Spring-AOP @AspectJ进阶之绑定连接点方法入参

概述


我们前面的博文在讲解切点函数时说过args()、this()、target()、@args()、@within()、@target()和@annotation()这7个函数除了可以指定类名外,还可以指定参数名将目标对象连接点上的方法入参绑定到增强的方法中。


其中args()用于绑定连接点方法的入参,@annotation()用于绑定连接点方法的注解对象,而@args()用于绑定连接点方法入参的注解


实例


来看一个args()绑定参数的实例

代码已托管到Github—> https://github.com/yangshangwei/SpringMaster


aHR0cDovL2ltZy5ibG9nLmNzZG4ubmV0LzIwMTcwOTEyMTQxMjQzNjg5.png

业务类

package com.xgj.aop.spring.advisor.aspectJAdvance.bindJointParameter;
import org.springframework.stereotype.Component;
/**
 * 
 * 
 * @ClassName: LogicBindService
 * 
 * @Description: @Component标注的Bean
 * 
 * @author: Mr.Yang
 * 
 * @date: 2017年9月12日 上午1:39:23
 */
@Component
public class LogicBindService {
  public void dealLogic(String bussiness, int number) {
    System.out.println("deal Logic:" + bussiness + ", number:" + number);
  }
}


编写切面

package com.xgj.aop.spring.advisor.aspectJAdvance.bindJointParameter;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
/**
 * 
 * 
 * @ClassName: BindJointPointParameterAspect
 * 
 * @Description: @Aspect标注的切面
 * 
 * @author: Mr.Yang
 * 
 * @date: 2017年9月12日 上午1:10:40
 */
@Aspect
public class BindJointPointParameterAspect {
  // ①绑定连接点参数,首先args(name,number,..)根据②处的增强方法入参找到name和number对应的类型,以得到真实的切点表达式:
  // target(com.xgj.aop.spring.advisor.aspectJAdvance.bindJointParameter.LogicBindService)
  // && args(String,int,..)
  // 在该增强方法织入到目标连接点时,增强方法可以通过num和name访问到连接点方法的入参。
  @Before("target(com.xgj.aop.spring.advisor.aspectJAdvance.bindJointParameter.LogicBindService) && args(name,number,..)")
  public void crossCodeCutting(int number, String name) throws Throwable { // ②增强方法接受连接点的参数
    System.out.println("----bindJoinPointParams()----");
    System.out.println("name:" + name);
    System.out.println("number:" + number);
    System.out.println("----bindJoinPointParams()----");
  }
}


在①处,我们通过args(name,number,…)进行连接点参数的绑定,和前面我们所讲述的方式不一样,当args()函数入参为参数名时,共包括两方面的信息:


连接点匹配规则信息:连接点方法第一个入参是String类型,第二个入参是int类型;


连接点方法入参和增强方法入参的绑定信息:连接点方法的第一个入参绑定到增强方法的name参数上,第二个入参绑定到增强方法的number入参上。


切点匹配和参数绑定的过程是这样的:


首先args()根据参数名称在增强方法中查到名称相同的入参并获知对应的类型,这样就知道匹配连接点方法的入参类型。


其次连接点方法入参类型所在的位置则由参数名在args()函数中声明的位置决定。


args(name,number)只匹配第一个入参是String第二个入参是int的目标类方法,如LogicBindService.dealLogic(String bussiness, int number)而不匹配LogicBindService.dealLogic( int number,String bussiness)


切点匹配和参数绑定过程:


aHR0cDovL2ltZy5ibG9nLmNzZG4ubmV0LzIwMTcwOTEyMTQwMzAzNDk2.png


和args()一样,其它可以绑定连接点参数的切点函数(如@args()和target()等),当指定参数名时,就同时具有匹配切点和绑定参数的双重功能


将业务Bean和切面配置到配置文件中

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
  xmlns:aop="http://www.springframework.org/schema/aop"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xmlns:context="http://www.springframework.org/schema/context"
  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
    http://www.springframework.org/schema/context 
        http://www.springframework.org/schema/context/spring-context.xsd">
<!-- (1)声明Context命名空间以及Schema文件   (2)扫描类包以及应用注解定义的bean -->
<context:component-scan base-package="com.xgj.aop.spring.advisor.aspectJAdvance.bindJointParameter"/>
<!-- 基于@AspectJ切面的驱动器 -->
<aop:aspectj-autoproxy proxy-target-class="true"/>
<!-- 使用了@AspectJ注解的切面类 -->
<bean class="com.xgj.aop.spring.advisor.aspectJAdvance.bindJointParameter.BindJointPointParameterAspect"/>
</beans>


测试类

package com.xgj.aop.spring.advisor.aspectJAdvance.bindJointParameter;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class BindJointPointParameterAspectTest {
  @Test
  public void test() {
    ApplicationContext ctx = new ClassPathXmlApplicationContext(
        "classpath:com/xgj/aop/spring/advisor/aspectJAdvance/bindJointParameter/conf-joinPoint.xml");
    LogicBindService logicBindService = ctx.getBean("logicBindService",
        LogicBindService.class);
    logicBindService.dealLogic("PROGRAMMING", 5);
  }
}


运行结果


2017-09-12 02:05:35,991  INFO [main] (AbstractApplicationContext.java:583) - Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@24177697: startup date [Tue Sep 12 02:05:35 BOT 2017]; root of context hierarchy
2017-09-12 02:05:36,098  INFO [main] (XmlBeanDefinitionReader.java:317) - Loading XML bean definitions from class path resource [com/xgj/aop/spring/advisor/aspectJAdvance/bindJointParameter/conf-joinPoint.xml]
----bindJoinPointParams()----
name:PROGRAMMING
number:5
----bindJoinPointParams()----
deal Logic:PROGRAMMING, number:5


可见,增强方法按预期绑定了LogicBindService.dealLogic(String bussiness, int number)方法的运行期入参。


Note: 为了保证实例能成功执行,必须启用CGLib动态代理:<aop:aspectj-autoproxy proxy-target-class="true" />,因为该实例需要对NaiveWaiter类进行代理(因为NaiveWaiter#simle()方法不是Waiter接口的方法),所以必须使用CGLib生成子类的代理方法。


提示 :为了保证实例能成功执行,必须启用CGLib动态代理:<aop:aspectj-autoproxy proxy-target-class="true" />,因为该实例不涉及到接口,所以必须使用CGLib生成子类的代理方法。 当然了,即使不设置(默认为jdk反向代理),当涉及的业务类没有接口时,spring会自动使用cglib代理

相关文章
|
19天前
|
Java Spring
【Spring】方法注解@Bean,配置类扫描路径
@Bean方法注解,如何在同一个类下面定义多个Bean对象,配置扫描路径
150 73
|
2月前
|
前端开发 Java Spring
Spring MVC源码分析之DispatcherServlet#getHandlerAdapter方法
`DispatcherServlet`的 `getHandlerAdapter`方法是Spring MVC处理请求的核心部分之一。它通过遍历预定义的 `HandlerAdapter`列表,找到适用于当前处理器的适配器,并调用适配器执行具体的处理逻辑。理解这个方法有助于深入了解Spring MVC的工作机制和扩展点。
42 1
|
3月前
|
存储 安全 Java
|
2月前
|
前端开发 Java Spring
Spring MVC源码分析之DispatcherServlet#getHandlerAdapter方法
`DispatcherServlet`的 `getHandlerAdapter`方法是Spring MVC处理请求的核心部分之一。它通过遍历预定义的 `HandlerAdapter`列表,找到适用于当前处理器的适配器,并调用适配器执行具体的处理逻辑。理解这个方法有助于深入了解Spring MVC的工作机制和扩展点。
39 1
|
2月前
|
前端开发 Java Spring
Spring MVC源码分析之DispatcherServlet#getHandlerAdapter方法
`DispatcherServlet`的 `getHandlerAdapter`方法是Spring MVC处理请求的核心部分之一。它通过遍历预定义的 `HandlerAdapter`列表,找到适用于当前处理器的适配器,并调用适配器执行具体的处理逻辑。理解这个方法有助于深入了解Spring MVC的工作机制和扩展点。
40 0
|
4月前
Micronaut AOP与代理机制:实现应用功能增强,无需侵入式编程的秘诀
AOP(面向切面编程)能够帮助我们在不修改现有代码的前提下,为应用程序添加新的功能或行为。Micronaut框架中的AOP模块通过动态代理机制实现了这一目标。AOP将横切关注点(如日志记录、事务管理等)从业务逻辑中分离出来,提高模块化程度。在Micronaut中,带有特定注解的类会在启动时生成代理对象,在运行时拦截方法调用并执行额外逻辑。例如,可以通过创建切面类并在目标类上添加注解来记录方法调用信息,从而在不侵入原有代码的情况下增强应用功能,提高代码的可维护性和可扩展性。
92 1
|
2月前
|
安全 Java 编译器
什么是AOP面向切面编程?怎么简单理解?
本文介绍了面向切面编程(AOP)的基本概念和原理,解释了如何通过分离横切关注点(如日志、事务管理等)来增强代码的模块化和可维护性。AOP的核心概念包括切面、连接点、切入点、通知和织入。文章还提供了一个使用Spring AOP的简单示例,展示了如何定义和应用切面。
253 1
什么是AOP面向切面编程?怎么简单理解?
|
2月前
|
XML Java 开发者
论面向方面的编程技术及其应用(AOP)
【11月更文挑战第2天】随着软件系统的规模和复杂度不断增加,传统的面向过程编程和面向对象编程(OOP)在应对横切关注点(如日志记录、事务管理、安全性检查等)时显得力不从心。面向方面的编程(Aspect-Oriented Programming,简称AOP)作为一种新的编程范式,通过将横切关注点与业务逻辑分离,提高了代码的可维护性、可重用性和可读性。本文首先概述了AOP的基本概念和技术原理,然后结合一个实际项目,详细阐述了在项目实践中使用AOP技术开发的具体步骤,最后分析了使用AOP的原因、开发过程中存在的问题及所使用的技术带来的实际应用效果。
77 5
|
4月前
Micronaut AOP与代理机制:实现应用功能增强,无需侵入式编程的秘诀
【9月更文挑战第9天】AOP(面向切面编程)通过分离横切关注点提高模块化程度,如日志记录、事务管理等。Micronaut AOP基于动态代理机制,在应用启动时为带有特定注解的类生成代理对象,实现在运行时拦截方法调用并执行额外逻辑。通过简单示例展示了如何在不修改 `CalculatorService` 类的情况下记录 `add` 方法的参数和结果,仅需添加 `@Loggable` 注解即可。这不仅提高了代码的可维护性和可扩展性,还降低了引入新错误的风险。
55 13
|
3月前
|
Java 容器
AOP面向切面编程
AOP面向切面编程
53 0