4

编辑

我创建了一个复制问题的测试项目。它可以在https://github.com/tomverelst/test-batch找到。

首先运行maven命令exec:java启动一个HSQL数据库。然后您可以运行 JUnit 测试MigrationJobConfigurationTest来加载 Spring 应用程序上下文。

原始问题

启动我的 Spring Batch 应用程序时,当 Spring 加载我的作业配置时出现以下异常:

Caused by: org.springframework.aop.framework.AopConfigException: Could not generate CGLIB subclass of class [class com.sun.proxy.$Proxy34]: Common causes of this problem include using a final class or a non-visible class; nested exception is java.lang.IllegalArgumentException: Cannot subclass final class class com.sun.proxy.$Proxy34

这是由@StepScope我的作业配置中的注释引起的。它试图用CGLIB 代理一个已经用JDK 代理代理的类,我不知道这个JDK 代理来自哪里。

我也尝试过使用@Scope(value = "step", proxyMode = ScopedProxyMode.NO),但是在调用 JDK 代理时出现堆栈溢出错误,该代理不断调用自身。

如果我删除@StepScope注释,应用程序将正确启动,但我需要能够将它们用于我的工作。

弹簧配置

<context:component-scan base-package="com.jnj.rn2.batch" />

<context:annotation-config />

<aop:aspectj-autoproxy proxy-target-class="true" />

<bean class="org.springframework.batch.core.scope.StepScope" />

// Job repository etc
...

迁移作业配置

@Configuration
public class MigrationJobConfiguration {

    @Autowired
    private JobBuilderFactory jobs;

    @Autowired
    private StepBuilderFactory steps;

    @Autowired
    private MigrationService migrationService;

    @Bean
    public Job migrationJob() {
        return jobs.get( "migrationJob" )
            .start( migrateCrfStep() )
            .next( indexRequestsStep() )
            .build();
    }

    @Bean
    public Step migrateCrfStep() {
        return steps.get( "migrateCrfStep" )
            .tasklet( migrateCrfTasklet() )
            .build();
    }

    @Bean
    public Step indexRequestsStep() {
        return steps.get( "indexRequestsStep" )
            .<LegacyRequest,LegacyRequest> chunk( 5 )
            .reader( indexRequestReader() )
            .processor( indexRequestProcessor() )
            .writer( indexRequestWriter() )
            .build();
    }

    @Bean
    @StepScope
    public MigrateCrfTasklet migrateCrfTasklet() {
        return new MigrateCrfTasklet();
    }

    @Bean
    @StepScope
    public IndexRequestItemReader indexRequestReader() {
        return new IndexRequestItemReader();
    }

    @Bean
    @StepScope
    public IndexRequestItemProcessor indexRequestProcessor() {
        return new IndexRequestItemProcessor();
    }

    @Bean
    @StepScope
    public IndexRequestItemWriter indexRequestWriter() {
        return new IndexRequestItemWriter();
    }

    // Setters
    ...
}
4

2 回答 2

7

我不能(还)给你一个实际的答案,但是我已经调试了您提供的示例,并且正在发生这种情况:

  • @Configuration定义阅读器看到@StepScope带有注释的注释@Scope(value = "step", proxyMode = ScopedProxyMode.TARGET_CLASS)
  • 读取器的 CGLIB 子类创建并注册为reader,而原始 bean 注册为scopedTarget.reader.
  • StepScope启动和后处理step作用域 bean。它检测到 CGLIB 扩展reader定义并尝试为该 => ERROR创建代理。

有两种代理机制存在冲突。发生了一些非常奇怪的事情,在我看来,这永远行不通。将尝试通过 Spring JIRA 进行搜索。


更新

找到解决方案 - 您需要禁用步骤范围的自动代理:

<bean class="org.springframework.batch.core.scope.StepScope">
    <property name="autoProxy" value="false" />
</bean>
于 2013-10-06T12:30:46.950 回答
1

轻松<bean class="org.springframework.batch.core.scope.StepScope" />从您的配置文件中删除;您不需要显式添加它,因为它是在批处理命名空间中定义的(如Step scope 的官方文档中所述)

于 2013-10-04T06:36:18.587 回答