java - 在 Spring AOP 或 AspectJ 中拦截带注释的类和方法

标签 java spring-boot annotations aspectj spring-aop

所以我有一个自定义注释

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface Intercepted {}

我想用它来将方面编织到方法中(AspectJ,@annotation(Intercepted))。

这个想法是,当我直接注释方法 @Intercepted 时,我会编织方面 - 该部分有效 - 或者如果我注释类,则应将方面编织到其所有 ( public) 方法——那部分没有。

此外,如果我注释一个类及其方法之一,则该方面应该只编织一次,方法级注释将覆盖类级注释。

本质上,我想要“如果存在类级注释,则添加类级注释,但前提是还没有方法级注释。”

我该怎么做?

最佳答案

这是一个 AspectJ 示例。 Spring AOP 中的切入点语法是相同的。

帮助类:

package de.scrum_master.app;

import java.lang.annotation.*;

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface Intercepted {}
package de.scrum_master.app;

@Intercepted
public class AnnotatedClass {
  public void doSomething() {}
  public void doSomethingElse() {}
}
package de.scrum_master.app;

public class AnnotatedMethod {
  @Intercepted
  public void doSomething() {}
  public void doSomethingElse() {}
}
package de.scrum_master.app;

@Intercepted
public class AnnotatedMixed {
  @Intercepted
  public void doSomething() {}
  public void doSomethingElse() {}
}

驱动程序应用程序(Java SE,无 Spring):

package de.scrum_master.app;

public class Application {
  public static void main(String[] args) {
    // Should be logged
    new AnnotatedClass().doSomething();
    // Should be logged
    new AnnotatedClass().doSomethingElse();

    // Should be logged
    new AnnotatedMethod().doSomething();
    // Should NOT be logged
    new AnnotatedMethod().doSomethingElse();

    // Should be logged, but only once
    new AnnotatedMixed().doSomething();
    // Should be logged
    new AnnotatedMixed().doSomethingElse();
  }
}

方面:

请注意execution(* *(..)) &&部分在 Spring AOP 中不是必需的,因为那里只支持方法执行连接点。切入点可以是 annotatedMethod() || annotatedClass()那里。在 AspectJ 中,我必须更加精确,因为否则会记录其他连接点类型。

package de.scrum_master.aspect;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;

@Aspect
public class AnnotationInterceptor {
  @Pointcut("@annotation(de.scrum_master.app.Intercepted)")
  public void annotatedMethod() {}

  @Pointcut("@within(de.scrum_master.app.Intercepted)")
  public void annotatedClass() {}

  @Before("execution(* *(..)) && (annotatedMethod() || annotatedClass())")
  public void log(JoinPoint thisJoinPoint) {
    System.out.println(thisJoinPoint);
  }
}

控制台日志:

execution(void de.scrum_master.app.AnnotatedClass.doSomething())
execution(void de.scrum_master.app.AnnotatedClass.doSomethingElse())
execution(void de.scrum_master.app.AnnotatedMethod.doSomething())
execution(void de.scrum_master.app.AnnotatedMixed.doSomething())
execution(void de.scrum_master.app.AnnotatedMixed.doSomethingElse())

关于java - 在 Spring AOP 或 AspectJ 中拦截带注释的类和方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56455428/

相关文章:

java - hibernate 。我们如何使用注释设置 PK/FK 的名称?

java - 如何使用带注释的对象而不是 hbms 从 Hibernate 角度执行 Hibernate 查询?

java - 数据包损坏和 UDP

spring-boot - Camunda 网络应用程序拒绝在本地主机上连接

java - JedisCluster : redis. clients.jedis.exceptions.JedisNoReachableClusterNodeException:集群中没有可访问的节点

java - Spring应用程序未启动

java - 在 JAVA 和 Spring 中设计自定义工作流程

java - Hibernate 自定义查找器 SQL 查询不返回 OneToMany 关系

java - java.awt.Color 对象的大小

java - 从 R.raw 文件夹中打开文件