java - Advice 从具体非泛型类中的泛型抽象类继承方法

标签 java spring aop spring-aop

例如我有如下界面

public interface Converter<I, O> {
    public O convert(I input);
}

实现该接口(interface)的抽象类

public abstract class AbstractIntegerConverter<T> implements Converter<T, Integer> {

    public Integer convert(T input) {
        // convert anything to int
    }

    public abstract Integer defaultValue();
}

及具体实现

@Component
public class StringToIntegerConverter extends AbstractIntegerConverter<String> {

    @Override
    public Integer defaultValue() {
        // implementation
    }
}

我想建议所有将 String 转换为任何内容的方法。我创建了以下方面

@Aspect
@Component
public class ConverterAspect {

    @Around("execution(* *..*.Converter+.convert(String))")
    public Object adviceStringConverter(ProceedingJoinPoint joinPoint) throws Throwable {
        // logic
    }
}

这是行不通的。 Spring 不会为 StringToIntegerConverter 类创建代理。 但是,如果我重写抽象类中的 convert(String input) 方法,它将开始工作并且 Spring 成功地为 StringToIntegerConverter 创建代理并执行所有需要的逻辑。

@Component
public class StringToIntegerConverter extends AbstractIntegerConverter<String> {

    @Override
    public Integer convert(String input) {
        return super.convert(input);
    }

    @Override
    public Integer defaultValue() {
        // implementation
    }
}

为什么会这样?有什么方法可以定义切入点,这样我就不需要重写 convert(String input) 方法了吗?

最佳答案

AbstractIntegerConverter 中,方法签名不是 public Integer convert(String input) 而是 public Integer convert(T input),因此您的切入点* *..*.Converter+.convert(String) 不匹配。改为使用 Object* 并通过 args() 验证运行时类型,而不是使用编译时签名:

@Around("execution(* *..Converter+.convert(*)) && args(input)")
public Object adviceStringConverter(ProceedingJoinPoint joinPoint, String input) throws Throwable {
    System.out.println(joinPoint);
    return joinPoint.proceed();
}

使用我添加的日志行,您将在控制台上看到如下内容:

execution(Integer de.scrum_master.app.AbstractIntegerConverter.convert(Object))

看到了吗? convert(Object),而不是 convert(String),因此不匹配。

P.S.:这是一个有趣的问题,不像大多数 AspectJ 或 Spring AOP 问题那么无聊。所以谢谢你。 :-)


更新:

关于您的后续问题,这是 javap -s 打印的内容(方面已停用):

javap -s AbstractIntegerConverter.class

Compiled from "AbstractIntegerConverter.java"
public abstract class de.scrum_master.app.AbstractIntegerConverter<T> implements de.scrum_master.app.Converter<T, java.lang.Integer> {
  public de.scrum_master.app.AbstractIntegerConverter();
    descriptor: ()V

  public java.lang.Integer convert(T);
    descriptor: (Ljava/lang/Object;)Ljava/lang/Integer;

  public abstract java.lang.Integer defaultValue();
    descriptor: ()Ljava/lang/Integer;

  public java.lang.Object convert(java.lang.Object);
    descriptor: (Ljava/lang/Object;)Ljava/lang/Object;
}

看到了吗?签名是 convert(Object) 正如我方面的日志输出也显示的那样。

javap -s StringToIntegerConverter.class

Compiled from "StringToIntegerConverter.java"
public class de.scrum_master.app.StringToIntegerConverter extends de.scrum_master.app.AbstractIntegerConverter<java.lang.String> {
  public de.scrum_master.app.StringToIntegerConverter();
    descriptor: ()V

  public java.lang.Integer defaultValue();
    descriptor: ()Ljava/lang/Integer;
}

具体转换器类中没有convert(String)convert(whatever) 方法。

你现在相信我了吗?

关于java - Advice 从具体非泛型类中的泛型抽象类继承方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54315147/

相关文章:

javascript - 定时自动更新 JSP 页面

java - 使用 JavaParser 将字符串字段添加到新的compilationUnit

java - 从现有 Java 代码生成 Swagger 文档?

java - 从aspectJ中带注释的方法获取局部变量值

Spring 依赖注入(inject)或切面编程

java - BindingResult 和 bean 名称 'searchPhrase' 的普通目标对象都不能作为请求属性

java - 如何在 Hibernate 中使用列表关联获取对象

java - Spring 批处理作业状态配置为使用 prometheus 发出警报

java - 有没有一种方法可以仅使用注释从同一个类声明两个 spring bean 实例?

java - Spring 声明回滚的例子?