@within @annotion @target 的区别

http://blog.youkuaiyun.com/liuwei8nn1/article/details/62216336

自从使用AspectJ风格切面配置,使得Spring的切面配置大大简化,但是AspectJ是另外一个开源项目,其规则表达式的语法也稍稍有些怪异。
 
下面给出一些常见示例的写法:
 
比如,下面是一个对Service包上所有方法的切面配置
         < aop:config > 
                 < aop:pointcut  id ="serviceOperation" 
                                             expression ="execution(* *..service*..*(..))" /> 
                 < aop:advisor  pointcut-ref ="serviceOperation" 
                                          advice-ref ="txAdvice" /> 
         </ aop:config >
 
表达式所处位置如上pointcut的位置。
 
配置这个为了更好控制切面上的事务,下面是一个事物配置的简单例子:
         < tx:advice  id ="txAdvice"  transaction-manager ="transactionManager" > 
                 < tx:attributes > 
                         < tx:method  name ="delete*"  rollback-for ="Exception" /> 
                         < tx:method  name ="save*"  rollback-for ="Exception" /> 
                         < tx:method  name ="update*"  rollback-for ="Exception" /> 
                         < tx:method  name ="*"  read-only ="true"  rollback-for ="Exception" /> 
                 </ tx:attributes > 
         </ tx:advice >
 
通过切面、通知的配置,就为所有的delete/save/update开头的方法添加上了一致性事务,对其他方法添加上了只读事务。
 
 
这个还不够细,如果要写更为详细的控制,就需要研究AspectJ切点配置的语法了,其实研究这些标准,还不如拿几个例子看看,解决实际问题就行了。就像写正则表达式一样,标准明摆着,要写好却不容易,从例子着手就能快速上手和领悟。
 
以下文档来自Spring中文开发指南2.5文档,由满江红开源组织翻译:
 
Spring AOP 用户可能会经常使用  execution切入点指示符。执行表达式的格式如下:
execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern)
          throws-pattern?)
除了返回类型模式(上面代码片断中的ret-type-pattern),名字模式和参数模式以外, 所有的部分都是可选的。返回类型模式决定了方法的返回类型必须依次匹配一个连接点。 你会使用的最频繁的返回类型模式是 *,它代表了匹配任意的返回类型。 一个全限定的类型名将只会匹配返回给定类型的方法。名字模式匹配的是方法名。 你可以使用 *通配符作为所有或者部分命名模式。 参数模式稍微有点复杂: ()匹配了一个不接受任何参数的方法, 而 (..)匹配了一个接受任意数量参数的方法(零或者更多)。 模式 (*)匹配了一个接受一个任何类型的参数的方法。 模式 (*,String)匹配了一个接受两个参数的方法,第一个可以是任意类型, 第二个则必须是String类型。更多的信息请参阅AspectJ编程指南中  语言语义的部分。
下面给出一些通用切入点表达式的例子。
  • 任意公共方法的执行:
    execution(public * *(..))
  • 任何一个名字以“set”开始的方法的执行:
    execution(* set*(..))
  • AccountService接口定义的任意方法的执行:
    execution(* com.xyz.service.AccountService.*(..))
  • 在service包中定义的任意方法的执行:
    execution(* com.xyz.service.*.*(..))
  • 在service包或其子包中定义的任意方法的执行:
    execution(* com.xyz.service..*.*(..))
  • 在service包中的任意连接点(在Spring AOP中只是方法执行):
    within(com.xyz.service.*)
  • 在service包或其子包中的任意连接点(在Spring AOP中只是方法执行):
    within(com.xyz.service..*)
  • 实现了 AccountService接口的代理对象的任意连接点 (在Spring AOP中只是方法执行):
    this(com.xyz.service.AccountService)
    'this'在绑定表单中更加常用:- 请参见后面的通知一节中了解如何使得代理对象在通知体内可用。
  • 实现 AccountService接口的目标对象的任意连接点 (在Spring AOP中只是方法执行):
    target(com.xyz.service.AccountService)
    'target'在绑定表单中更加常用:- 请参见后面的通知一节中了解如何使得目标对象在通知体内可用。
  • 任何一个只接受一个参数,并且运行时所传入的参数是 Serializable 接口的连接点(在Spring AOP中只是方法执行)
    args(java.io.Serializable)
    'args'在绑定表单中更加常用:- 请参见后面的通知一节中了解如何使得方法参数在通知体内可用。
    请注意在例子中给出的切入点不同于  execution(* *(java.io.Serializable)): args版本只有在动态运行时候传入参数是Serializable时才匹配,而execution版本在方法签名中声明只有一个  Serializable类型的参数时候匹配。
  • 目标对象中有一个  @Transactional 注解的任意连接点 (在Spring AOP中只是方法执行)
    @target(org.springframework.transaction.annotation.Transactional)
    '@target'在绑定表单中更加常用:- 请参见后面的通知一节中了解如何使得注解对象在通知体内可用。
  • 任何一个目标对象声明的类型有一个  @Transactional 注解的连接点 (在Spring AOP中只是方法执行):
    @within(org.springframework.transaction.annotation.Transactional)
    '@within'在绑定表单中更加常用:- 请参见后面的通知一节中了解如何使得注解对象在通知体内可用。
  • 任何一个执行的方法有一个  @Transactional 注解的连接点 (在Spring AOP中只是方法执行)
    @annotation(org.springframework.transaction.annotation.Transactional)
    '@annotation'在绑定表单中更加常用:- 请参见后面的通知一节中了解如何使得注解对象在通知体内可用。
  • 任何一个只接受一个参数,并且运行时所传入的参数类型具有 @Classified 注解的连接点(在Spring AOP中只是方法执行)
    @args(com.xyz.security.Classified)
    '@args'在绑定表单中更加常用:- 请参见后面的通知一节中了解如何使得注解对象在通知体内可用。
  • 任何一个在名为' tradeService'的Spring bean之上的连接点 (在Spring AOP中只是方法执行):
    bean(tradeService)
  • 任何一个在名字匹配通配符表达式' *Service'的Spring bean之上的连接点 (在Spring AOP中只是方法执行):
    bean(*Service)
<think>我们正在讨论AspectJ中的@within注解。根据提供的引用,特别是引用[4],我们可以看到@within在切面定义中的使用示例。 引用[4]展示了一个切面类HumanAspect,其中包含了使用@within的切点表达式。例如: @Before("@within(demon.study.spring5.spring_aop.sample.A1)") public void execute1(){...} 这里的@within(demon.study.spring5.spring_aop.sample.A1)表示匹配那些在类级别上标注了@A1注解的类中的所有方法。 因此,@within注解的作用是:匹配指定注解所标注的类中的所有方法(注意:不包括父类中的方法)。也就是说,如果一个类被指定的注解标记,那么这个类的所有方法都会被匹配。 使用场景:当我们想要对某个特定注解标记的类中的所有方法进行增强(如日志记录、事务管理等)时,可以使用@within。例如,我们可以自定义一个@Loggable注解,然后使用@within(@Loggable)来匹配所有被@Loggable注解标记的类的方法。 需要注意的是,@within@target区别: - @within:在匹配连接点时,只考虑目标对象的类上是否有指定的注解,而不考虑其父类(因为注解继承默认是不支持的,除非注解本身被@Inherited标记)。 - @target:在匹配连接点时,考虑目标对象上是否有指定的注解(即运行时对象的类上的注解,包括继承的注解,如果注解有@Inherited的话)。 引用[4]中的示例代码也展示了@target的使用,我们可以通过运行示例来观察它们的不同行为。 总结: @within注解用于匹配那些在类级别上(直接或间接,但注意注解继承性)有指定注解的类中的所有方法。 根据上述理解,我们可以回答用户的问题。 注意:在AspectJ中,注解匹配还有其他的方式,如@annotation用于匹配方法上带有指定注解的连接点。 下面我们按照要求,用中文回答,并生成相关问题。</think>### AspectJ中@within注解的作用与使用场景 #### 作用说明 `@within`是AspectJ中用于**类级别注解匹配**的切点指示符(Pointcut Designator)。其核心作用是: 1. **匹配规则**:匹配所有**直接或间接带有指定注解的类**中的方法执行连接点 $$ \text{@within(注解类型)} $$ 2. **作用范围**:当类声明中包含目标注解时,该类内部的所有方法(包括构造方法)都会被匹配 3. **继承特性**:仅匹配当前类声明的注解,**不包含父类继承的注解**(与`@target`的关键区别)[^4] #### 使用场景示例 ```java // 自定义注解 @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.TYPE) public @interface AuditLog {} // 类级审计日志注解 // 切面定义 @Aspect @Component public class AuditAspect { // 匹配所有被@AuditLog注解的类中的方法 @Before("@within(com.example.AuditLog)") public void logMethodAccess(JoinPoint jp) { System.out.println("审计访问: " + jp.getSignature()); } } // 使用示例 @AuditLog // 触发切面匹配 @Service public class UserService { public void createUser() { ... } // 被增强方法 public void deleteUser() { ... } // 被增强方法 } ``` #### 与@target区别 | 特性 | @within | @target | |------------|--------------------------------|--------------------------------| | **匹配维度** | 类声明上的注解 | 目标对象运行时类型的注解 | | **继承处理** | 不匹配继承的注解 | 匹配继承的注解(若注解可继承)| | **性能** | 编译期织入效率更高 | 需运行时检查 | | **典型场景** | 精确控制类级别的横切逻辑 | 需要处理子类继承注解的场景 | > 引用示例中的`@Before("@within(demon.study...A1)")`正是基于类注解匹配的典型实现[^4] #### 实际应用场景 1. **统一日志审计**:为标注`@Audit`的类自动添加操作日志 2. **事务分组管理**:对标注`@TransactionalService`的类启用特殊事务策略 3. **安全控制**:拦截`@SecuredComponent`类中的所有方法进行权限校验 4. **性能监控**:对标注`@Monitor`的类进行方法执行耗时统计 5. **DTO验证**:自动校验带有`@Validatable`注解的类中所有方法的参数 > 注:Spring AOP与AspectJ协同使用时,需确保注解具有`RUNTIME`保留策略[^1][^2]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值