@Pointcut 使用@annotation 带参数

本文深入探讨了AOP(面向切面编程)的基本概念,包括Advice(通知)、JointPoint(连接点)、Pointcut(切入点)、Advisor(增强)及AOPProxy(AOP代理)。详细解释了@Before、@After、@AfterReturning、@AfterThrowing和@Around等通知类型,并介绍了如何使用Spring配置AOP,包括切面、切入点表达式和通知的定义。
AOP的基本概念
  1. Advice(通知、切面): 某个连接点所采用的处理逻辑,也就是向连接点注入的代码, AOP在特定的切入点上执行的增强处理。

    1.1 @Before: 标识一个前置增强方法,相当于BeforeAdvice的功能.
    1.2 @After: final增强,不管是抛出异常或者正常退出都会执行.
    1.3 @AfterReturning: 后置增强,似于AfterReturningAdvice, 方法正常退出时执行.
    1.4 @AfterThrowing: 异常抛出增强,相当于ThrowsAdvice.
    1.5 @Around: 环绕增强,相当于MethodInterceptor.

  2. JointPoint(连接点):程序运行中的某个阶段点,比如方法的调用、异常的抛出等。

  3. Pointcut(切入点): JoinPoint的集合,是程序中需要注入Advice的位置的集合,指明Advice要在什么样的条件下才能被触发,在程序中主要体现为书写切入点表达式。

  4. Advisor(增强): 是PointCut和Advice的综合体,完整描述了一个advice将会在pointcut所定义的位置被触发。

  5. @Aspect(切面): 通常是一个类的注解,里面可以定义切入点和通知

  6. AOP Proxy:AOP框架创建的对象,代理就是目标对象的加强。Spring中的AOP代理可以使JDK动态代理,也可以是CGLIB代理,前者基于接口,后者基于子类。

<aop:aspectj-autoproxy/>
   <aop:config proxy-target-class="true">
       <aop:pointcut id="servicePointcut"
                     expression="execution(* com.cpic..*Service.*(..))" />
       <aop:advisor pointcut-ref="servicePointcut" advice-ref="txAdvice"
                    order="3" />
   </aop:config>
   <tx:advice id="txAdvice" transaction-manager="transactionManager">
       <tx:attributes>
           <tx:method name="list*" read-only="true" />
           <!-- log方法会启动一个新事务 -->
           <tx:method name="log*" propagation="REQUIRES_NEW"
                      isolation="READ_COMMITTED" />
       </tx:attributes>
   </tx:advice>

//OK所以一个Spring增强(advisor)=切面(advice)+切入点(PointCut)
Pointcut

表示式(expression)和签名(signature)

//Pointcut表示式
@Pointcut("execution(* com.savage.aop.MessageSender.*(..))")
//Point签名
private void log(){} 

由下列方式来定义或者通过 &&、 ||、 !、 的方式进行组合:

  • execution:用于匹配方法执行的连接点;
  • within:用于匹配指定类型内的方法执行;
  • this:用于匹配当前AOP代理对象类型的执行方法;注意是AOP代理对象的类型匹配,这样就可能包括引入接口也类型匹配;
  • target:用于匹配当前目标对象类型的执行方法;注意是目标对象的类型匹配,这样就不包括引入接口也类型匹配;
  • args:用于匹配当前执行的方法传入的参数为指定类型的执行方法;
  • @within:用于匹配所以持有指定注解类型内的方法;
  • @target:用于匹配当前目标对象类型的执行方法,其中目标对象持有指定的注解;
  • @args:用于匹配当前执行的方法传入的参数持有指定注解的执行;
  • @annotation:用于匹配当前执行方法持有指定注解的方法;
格式
execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern)throws-pattern?) 

其中后面跟着“?”的是可选项括号中各个pattern分别表示:

  • 修饰符匹配(modifier-pattern?)
  • 返回值匹配(ret-type-pattern): 可以为*表示任何返回值, 全路径的类名等
  • 类路径匹配(declaring-type-pattern?)
  • 方法名匹配(name-pattern):可以指定方法名 或者 代表所有, set 代表以set开头的所有方法
  • 参数匹配((param-pattern)):可以指定具体的参数类型,多个参数间用“,”隔开,各个参数也可以用"" 来表示匹配任意类型的参数,"…"表示零个或多个任意参数。
    如(String)表示匹配一个String参数的方法;(
    ,String) 表示匹配有两个参数的方法,第一个参数可以是任意类型,而第二个参数是String类型。
  • 异常类型匹配(throws-pattern?)
eg.
  • 任意公共方法的执行:execution(public * *(…))
  • 任何一个以“set”开始的方法的执行:execution(* set*(…))
  • AccountService 接口的任意方法的执行:execution(* com.xyz.service.AccountService.*(…))
  • 定义在service包里的任意方法的执行: execution(* com.xyz.service..(…))
  • 定义在service包和所有子包里的任意类的任意方法的执行:execution(* com.xyz.service….(…))
    第一个表示匹配任意的方法返回值, …(两个点)表示零个或多个,第一个…表示service包及其子包,第二个表示所有类, 第三个*表示所有方法,第二个…表示方法的任意参数个数
  • 定义在pointcutexp包和所有子包里的JoinPointObjP2类的任意方法的执行:execution(* com.test.spring.aop.pointcutexp…JoinPointObjP2.*(…))")
  • pointcutexp包里的任意类: within(com.test.spring.aop.pointcutexp.*)
  • pointcutexp包和所有子包里的任意类:within(com.test.spring.aop.pointcutexp…*)
  • 实现了Intf接口的所有类,如果Intf不是接口,限定Intf单个类:this(com.test.spring.aop.pointcutexp.Intf)
  • 当一个实现了接口的类被AOP的时候,用getBean方法必须cast为接口类型,不能为该类的类型
  • 带有@Transactional标注的所有类的任意方法:
    @within(org.springframework.transaction.annotation.Transactional)
    @target(org.springframework.transaction.annotation.Transactional)
  • 带有@Transactional标注的任意方法:@annotation(org.springframework.transaction.annotation.Transactional)
    @within和@target针对类的注解,@annotation是针对方法的注解
  • 参数带有@Transactional标注的方法:@args(org.springframework.transaction.annotation.Transactional)
  • 参数为String类型(运行是决定)的方法: args(String)

转自:
https://my.oschina.net/u/3434392/blog/1625493
作者:noob_fly

<think>好的,我现在需要帮助用户了解AspectJ中@Pointcut使用@within和@annotation的所有编写方式,并提供示例代码。首先,我得回顾一下用户的问题和提供的引用内容,确保我正确理解他们的需求。 用户之前的问题是关于@Pointcut使用@within和@annotation的所有实现方式,需要示例代码。引用的内容中提到了多个相关的知识点,比如@within用于匹配持有指定注解的类型内的方法,@annotation用于匹配带有指定注解的方法。引用4和5特别详细说明了这些指示符的用法,还有示例代码如引用3和5中的切点定义。 首先,我需要整理@within和@annotation的基本语法和用法。根据引用4,@within匹配的是类级别的注解,即该类的所有方法都会被拦截,而@annotation则是方法级别的注解,只有带有该注解的方法才会被拦截。此外,引用5中的例子展示了如何用||组合这两个指示符,这可能也是用户需要的一个点。 接下来,我需要考虑用户可能需要的所有编写方式。这包括单独使用@within和@annotation,以及它们的组合使用。例如,单独使用@annotation来拦截方法,或者单独使用@within来拦截类中的所有方法。另外,可能还需要考虑参数传递的情况,比如在切点表达式中如何指定注解类型,以及如何引用其他切点。 然后,示例代码部分需要覆盖这些情况。比如,单独使用@annotation的示例,单独使用@within的示例,以及它们的组合使用。同时,要注意代码的正确性,确保注解的路径正确,切点表达式语法正确,并且通知方法正确绑定参数。例如,引用3中的示例使用了execution指示符,而用户需要的是@within和@annotation,所以需要调整。 还需要注意AspectJ和Spring AOP的区别。根据引用2,虽然使用了AspectJ的注解,但实现是Spring AOP自己的,因此某些高级特性可能不支持,但用户的问题集中在@within和@annotation,这两个在Spring AOP中是支持的。 另外,用户可能需要了解如何在实际应用中组合不同的指示符,比如使用逻辑运算符&&或者||,如引用5中的例子使用了||来组合@annotation和@within。这也是一个重要的点,应该包含在回答中。 现在,我需要组织这些信息,分点说明@within和@annotation的各种使用方式,每个点附上示例代码,并解释其作用。同时,需要确保代码中的包路径和注解名称是示例性质的,如使用自定义的@MyClassAnnotation和@MyMethodAnnotation,这样用户更容易理解。 最后,生成的相关问题需要与主题相关,帮助用户进一步深入学习,比如关于执行顺序、参数获取、组合切点等,这些都是在实际开发中可能遇到的问题。</think>### AspectJ @Pointcut 使用 @within 和 @annotation 的编写方式详解 #### 一、@within 注解的典型用法 **匹配带有指定类注解的所有方法** ```java @Pointcut("@within(com.example.MyClassAnnotation)") public void withinAnnotationPointcut() {} ``` - **应用场景**:拦截所有被`@MyClassAnnotation`标注的类中的**所有方法** - **示例**:若类`UserService`标注了`@MyClassAnnotation`,则其中所有方法都会被拦截[^4][^5] #### 二、@annotation 注解的核心用法 **匹配带有指定方法注解的方法** ```java @Pointcut("@annotation(com.example.MyMethodAnnotation)") public void annotationPointcut() {} ``` - **应用场景**:精确拦截被`@MyMethodAnnotation`标注的**具体方法** - **示例**:方法`public void updateUser(@MyMethodAnnotation User user)`会被拦截[^2][^5] #### 三、组合使用 @within 和 @annotation **逻辑运算符实现复合条件** ```java @Pointcut("@within(com.example.MyClassAnnotation) || @annotation(com.example.MyMethodAnnotation)") public void combinedPointcut() {} ``` - **应用场景**:同时拦截类注解和方法注解标记的内容 - **效果**:满足类注解`@MyClassAnnotation`或方法注解`@MyMethodAnnotation`任一条件即触发[^5] #### 四、参数绑定的高级用法 **获取注解实例并传递到增强逻辑** ```java @Before("@within(classAnn) || @annotation(methodAnn)") public void beforeAdvice(JoinPoint jp, MyClassAnnotation classAnn, MyMethodAnnotation methodAnn) { // 可获取具体注解的属性值 } ``` - **注意**:需要确保通知方法的参数名称与切点表达式中的标识符一致[^4] #### 五、完整切面实现示例 ```java @Aspect @Component public class CustomAspect { // 类注解切点 @Pointcut("@within(com.example.MyClassAnnotation)") public void classAnnotationPointcut() {} // 方法注解切点 @Pointcut("@annotation(com.example.MyMethodAnnotation)") public void methodAnnotationPointcut() {} // 组合切点 @Pointcut("classAnnotationPointcut() || methodAnnotationPointcut()") public void combinedPointcut() {} @Around("combinedPointcut()") public Object process(ProceedingJoinPoint pjp) throws Throwable { MethodSignature signature = (MethodSignature) pjp.getSignature(); // 获取方法上的注解 MyMethodAnnotation methodAnn = signature.getMethod().getAnnotation(MyMethodAnnotation.class); // 获取类上的注解 MyClassAnnotation classAnn = pjp.getTarget().getClass().getAnnotation(MyClassAnnotation.class); System.out.println("拦截到增强方法: " + signature.getName()); return pjp.proceed(); } } ``` #### 六、特殊场景处理 **1. 嵌套类注解匹配** ```java @Pointcut("@within(com.example..*)") // 包通配符匹配 public void packageWildcard() {} ``` **2. 注解继承支持** ```java @Pointcut("@within(org.springframework.stereotype.Service)") public void springServiceAnnotation() {} ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值