java - 修改父类(super class)中的注释值并使用新值动态实例化子类

标签 java spring spring-annotations spring-cloud-stream

我们使用 Spring Cloud Stream 作为基于微服务的架构中事件消息传递的底层实现。我们希望更进一步,在我们的服务和 Spring Cloud Stream 库之间提供一个抽象层,以允许动态 channel 订阅,而无需在服务本身中添加太多样板配置代码。

最初的想法如下:

messaging-library 提供了一个 BaseHandler 抽象类,所有单独的服务都必须实现该类。特定服务的所有处理程序都希望使用相同的输入 channel ,尽管只会调用与要处理的事件类型相对应的输入 channel 。看起来如下:

public abstract class BaseEventHandler<T extends Event> {
    @StreamListener
    public abstract void handle(T event);
}

每个服务都提供自己的事件包,其中包含N个事件处理程序。有一些普通的 POJO 必须以编程方式实例化。这看起来如下:

public class ServiceEventHandler extends BaseEventHandler<ImportantServiceEvent> {

    @Override
    public void handle(ImportantServiceEvent event) {
        // todo stuff
    }
}

请注意,此时这些是简单的类,而不是 Spring bean,其中 ImportantServiceEvent 实现了 Event

我们的消息库会在启动时尽早进行扫描,并执行处理程序初始化。为此,需要完成以下步骤:

  • 我们扫描类路径中提供某种事件处理的所有可用包,并检索 BaseEventHandler 的所有子类。
  • 我们在子类的层次结构中检索@StreamListener注释,并将其值更改为该服务对应的输入 channel 。
  • 由于我们的处理程序可能需要与其他一些应用程序组件(存储库等)通信,因此我们使用 DefaultListableBeanFactory 将我们的处理程序实例化为单例,如下所示:

    val bean = beanFactory.createBean(eventHandlerClass, AutowireCapableBeanFactory.AUTOWIRE_BY_TYPE, true);
    beanFactory.registerSingleton(eventHandlerClass.getSimpleName(), bean);
    

此后,我们遇到了几个问题。

Spring Cloud Stream @StreamListener 注解不能被继承,因为它是一个方法注解。尽管如此,某些机制似乎能够在父级上找到它(因为注册了 StreamListenerAnnotationBeanPostProcessor),并尝试在初始化的 ServiceEventHandler 上执行后处理。我们的假设是 Spring Cloud Stream 使用类似 AnnotationElementUtils.findAllMergedAnnotations() 的内容。

因此,我们认为我们可能能够在子类的每次实例化之前更改基类的注释值。因此,我们认为虽然我们的 BaseEventHandler 会简单地获取一个新值,然后在初始化阶段结束时保持不变,但子类将使用正确的 channel 名称进行实例化在实例化时,因为我们不希望重新绑定(bind)。然而,情况并非如此,所使用的 @StreamListener 注释的值始终是基础上的值。

那么问题是:Spring Cloud Stream 能实现我们想要的功能吗?或者我们这里遇到的是一个简单的 Java 问题(似乎并非如此)? Spring Cloud Stream 团队是否预见到了这样的用例,我们的做法是否完全错误?

这个问题也发布在Spring Cloud Stream tracker上以防它可能有助于获得更多关注。

最佳答案

由于同一个人监控 SO 和 GitHub 问题,因此在这两个地方发帖是毫无意义的。 Stack Overflow 是提问的首选。

您应该能够对 BPP 进行子类化;它特别有这个扩展点:

/**
 * Extension point, allowing subclasses to customize the {@link StreamListener}
 * annotation detected by the postprocessor.
 *
 * @param originalAnnotation the original annotation
 * @param annotatedMethod the method on which the annotation has been found
 * @return the postprocessed {@link StreamListener} annotation
 */
protected StreamListener postProcessAnnotation(StreamListener originalAnnotation, Method annotatedMethod) {
    return originalAnnotation;
}

然后用你的定义覆盖bean定义

@Bean(name = STREAM_LISTENER_ANNOTATION_BEAN_POST_PROCESSOR_NAME)
public static StreamListenerAnnotationBeanPostProcessor streamListenerAnnotationBeanPostProcessor() {
    return new StreamListenerAnnotationBeanPostProcessor();
}

关于java - 修改父类(super class)中的注释值并使用新值动态实例化子类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50970269/

相关文章:

java - 优先级队列不按优先级插入东西

spring - 在 Glassfish4 上部署 Grails 2.5.1 应用程序时出错

java - 使用自定义 ClassPathResource 配置 Spring 4 PropertySourcesPlaceholderConfigurer

spring - Spring @Async取消并开始吗?

java - AWS SDK for S3 中的 TransferManager 是否在执行异步 I/O?

java - Android 4.2.2 Wifi-Direct adhoc 网络——访问隐藏的 android 方法来设置 SSID 和密码?

java - addMouseListener 不适用于标签

java - Spring MVC InternalResourceViewResolver - 无法显示我的页面

java - Spring注解、读取属性

java - 自定义注释不适用于 spring Beans