Spring Batch:org.springframework.batch.item.ReaderNotOpenException:阅读器必须打开才能读取

Posted

技术标签:

【中文标题】Spring Batch:org.springframework.batch.item.ReaderNotOpenException:阅读器必须打开才能读取【英文标题】:Spring Batch: org.springframework.batch.item.ReaderNotOpenException: Reader must be open before it can be read 【发布时间】:2014-07-13 21:37:59 【问题描述】:

我阅读了与 SO 相关的问题,但解决方案对我不起作用。

我得到了org.springframework.batch.item.ReaderNotOpenException: Reader must be open before it can be read 异常。

下面是我的配置:

@Bean
@StepScope
public ItemReader<Player> reader(@Value("#jobParameters[inputZipfile]") String inputZipfile) 
                final String [] header =  .. this part omitted for brevity ... ;
                FlatFileItemReader<Player> reader = new FlatFileItemReader<Player>();


                System.out.println("\t\t\t\t\t"+inputZipfile);

                reader.setResource(new ClassPathResource(inputZipfile));
                reader.setLineMapper(new DefaultLineMapper<Player>() 
                    setLineTokenizer(new DelimitedLineTokenizer() 
                        setNames( header );
                    );
                    setFieldSetMapper(new BeanWrapperFieldSetMapper<Player>() 
                        setTargetType(Player.class);
                    );
                );
                reader.setComments( header );
                return reader;


@Bean
@StepScope
public ItemProcessor<Player, PlayersStats> processor(@Value("#jobParameters[statType]") String statType,
                                                                 @Value("#jobParameters[season]") String season)
                PlayersStatsProcessor psp = new PlayersStatsProcessor();
                psp.setStatisticType( StatisticType.valueOf(statType) );
                psp.setSeason( season );
                return psp;



@Bean
@StepScope
public ItemWriter<PlayersStats> writer()
            return new CustomWriter();



@Bean
public Job generateStatisticsJob() 

        return this.jobs.get("generateStatisticsJob")
                .incrementer(new RunIdIncrementer())
                .start(processPlayerStats())
                //.end()
                .build();


@Bean
public Step processPlayerStats() 
           return this.steps.get("processPlayerStats")        
                        .<Player, PlayersStats> chunk(10)
                        .reader(reader(null))
                        .processor(processor(null,null))
                        .writer(writer())
                        .build();

inputZipFile 变量设置正确并且文件存在于驱动器上。 我签入了 FlatFileItemReader 代码,当未设置阅读器类的阅读器成员时,会发生 ReaderNotOpenException。阅读器成员在 doOpen 方法中设置。 看起来 doOpen 没有被调用。问题是为什么?

【问题讨论】:

但是 FlatFileItemReader 能够读取 zip 文件吗? inputZipFile 是剩余的。我正在传递 txt 文件。 【参考方案1】:

当我将阅读器 bean 的返回类型从 Item 更改为 FlatFileItemReader 时,该问题消失了。我仍然不清楚为什么这是一个问题,因为 chunk().reader() 接受 ItemReader 作为输入。我假设在底层有一些 AOP 魔法,它执行 FlatFileReader 初始化并按返回类型匹配。

【讨论】:

顺便说一句,Michael Minella 在这里解释了这个 AOP 魔法:***.com/questions/21241683/… 确实@StepScope 被定义为@Scope(value = "step", proxyMode = ScopedProxyMode.TARGET_CLASS)【参考方案2】:

既然你把reader放在StepScope,那么bean返回类型应该是实现类型FlatFileItemReader

@Bean
@StepScope
public FlatFileItemReader<Player> reader(@Value("#jobParameters[inputZipfile]") String inputZipfile) 
            ...
            return reader;

如果指定接口,则Spring代理只能访问接口ItemReader上指定的方法和注解,缺少重要注解。日志中还有一个警告(有错字):

2015-05-07 10:40:22,733 WARN  [main] org.springframework.batch.item.ItemReader is an interface.  The implementing class will not be queried for annotation based listener configurations.  If using @StepScope on a @Bean method, be sure to return the implementing class so listner annotations can be used.
2015-05-07 10:40:22,748 WARN  [main] org.springframework.batch.item.ItemReader is an interface.  The implementing class will not be queried for annotation based listener configurations.  If using @StepScope on a @Bean method, be sure to return the implementing class so listner annotations can be used. 

目前 Spring Boot Batch 示例也返回了 ItemReader,所以我想其他人也会遇到同样的问题。

【讨论】:

它对我有用。 1. 添加@StepScope 2. 对于返回类型为ItemReader、ItemWriter的每个ItemReader、ItemWriter方法,将返回类型更改为JdbcCursorItemReader、JdbcCursorItemWriter等实际实现类。另外,如果您有阅读器“关闭”的异常,请添加 @Bean(destroyMethod="")【参考方案3】:

因为ItemReader没有open方法,使用hte StepScope会根据返回类型创建一个代理类。返回ItemStreamReader也可以

【讨论】:

【参考方案4】:

我已通过以下方式修复它:

reader.open(new ExecutionContext());

【讨论】:

在挣扎了 1.5 天后,这实际上对我有用。 这是一个坏技巧,不是正确的解决方案。如果您允许 Spring 自动调用 reader.open(),他会自动调用:您的 @Bean 方法必须按照其他答案的建议返回实现类。 感谢这个 hack,我可以将 Chunk 步骤阅读器回收到 Tasklet 步骤中。【参考方案5】:

我认为你应该在 processPlayerStats() step bean 类中增加你的块大小,即从块(10)到块(100/更多可能)。

【讨论】:

【参考方案6】:

我定义的方法如下:

    @Bean
    @StepScope
    public ItemReader<BP> BPReader()
       
        JdbcCursorItemReader<BP> itemReader = new JdbcCursorItemReader<BP>();
        ...
        return itemReader;
    

我在方法中定义的类型是ItemReader,它是一个接口,返回类型是JdbcCursorItemReader,它是它的一个子类。通过将返回类型定义更改为 JdbcCursorItemReader 解决了我的问题

【讨论】:

【参考方案7】:

同样的问题。 将阅读器的返回类型更改为实际实现并添加到阅读器

implements ItemStream

帮我搞定了

【讨论】:

以上是关于Spring Batch:org.springframework.batch.item.ReaderNotOpenException:阅读器必须打开才能读取的主要内容,如果未能解决你的问题,请参考以下文章

spring 学习总结

Spring MVC 中的 ModelMap 与模型

Spring Boot -- actuator

java编程spring里org.springframework.web这个包是干什么的

spring security实现在线用户统计

如何获取 Spring Security SessionRegistry?