【发布时间】:2016-06-06 11:10:33
【问题描述】:
背景
我正在设计一个文件读取层,它可以读取分隔文件并将其加载到List 中。我决定使用 Spring Batch 是因为它提供了许多可扩展性选项,我可以根据文件的大小将它们用于不同的文件集。
要求
- 我想设计一个通用的 Job API,可以用来读取任何分隔文件。
- 应该有一个单独的作业结构用于解析每个分隔文件。例如,如果系统需要读取 5 个文件,就会有 5 个作业(每个文件一个)。 5 个作业彼此不同的唯一方式是它们将使用不同的
FieldSetMapper、列名、目录路径和其他缩放参数,例如commit-interval和throttle-limit。 - 此 API 的用户不需要配置 Spring 当系统中引入新的文件类型时,他自己的批处理作业、步骤、分块、分区等。
- 用户只需提供作业使用的
FieldsetMapper、commit-interval、throttle-limit以及每种类型文件的存放目录。 - 每个文件将有一个预定义目录。每个目录可以包含多个相同类型和格式的文件。
MultiResourcePartioner将用于查看目录内部。分区数 = 目录中的文件数。
我的要求是构建一个 Spring Batch 基础架构,它可以为我提供一份独特的工作,一旦我掌握了构成该工作的点点滴滴,我就可以启动它。
我的解决方案:
我创建了一个抽象配置类,它将被具体配置类扩展(每个文件将有 1 个具体类要读取)。
@Configuration
@EnableBatchProcessing
public abstract class AbstractFileLoader<T> {
private static final String FILE_PATTERN = "*.dat";
@Autowired
JobBuilderFactory jobs;
@Autowired
ResourcePatternResolver resourcePatternResolver;
public final Job createJob(Step s1, JobExecutionListener listener) {
return jobs.get(this.getClass().getSimpleName())
.incrementer(new RunIdIncrementer()).listener(listener)
.start(s1).build();
}
public abstract Job loaderJob(Step s1, JobExecutionListener listener);
public abstract FieldSetMapper<T> getFieldSetMapper();
public abstract String getFilesPath();
public abstract String[] getColumnNames();
public abstract int getChunkSize();
public abstract int getThrottleLimit();
@Bean
@StepScope
@Value("#{stepExecutionContext['fileName']}")
public FlatFileItemReader<T> reader(String file) {
FlatFileItemReader<T> reader = new FlatFileItemReader<T>();
String path = file.substring(file.indexOf(":") + 1, file.length());
FileSystemResource resource = new FileSystemResource(path);
reader.setResource(resource);
DefaultLineMapper<T> lineMapper = new DefaultLineMapper<T>();
lineMapper.setFieldSetMapper(getFieldSetMapper());
DelimitedLineTokenizer tokenizer = new DelimitedLineTokenizer(",");
tokenizer.setNames(getColumnNames());
lineMapper.setLineTokenizer(tokenizer);
reader.setLineMapper(lineMapper);
reader.setLinesToSkip(1);
return reader;
}
@Bean
public ItemProcessor<T, T> processor() {
// TODO add transformations here
return null;
}
@Bean
@JobScope
public ListItemWriter<T> writer() {
ListItemWriter<T> writer = new ListItemWriter<T>();
return writer;
}
@Bean
@JobScope
public Step readStep(StepBuilderFactory stepBuilderFactory,
ItemReader<T> reader, ItemWriter<T> writer,
ItemProcessor<T, T> processor, TaskExecutor taskExecutor) {
final Step readerStep = stepBuilderFactory
.get(this.getClass().getSimpleName() + " ReadStep:slave")
.<T, T> chunk(getChunkSize()).reader(reader)
.processor(processor).writer(writer).taskExecutor(taskExecutor)
.throttleLimit(getThrottleLimit()).build();
final Step partitionedStep = stepBuilderFactory
.get(this.getClass().getSimpleName() + " ReadStep:master")
.partitioner(readerStep)
.partitioner(
this.getClass().getSimpleName() + " ReadStep:slave",
partitioner()).taskExecutor(taskExecutor).build();
return partitionedStep;
}
/*
* @Bean public TaskExecutor taskExecutor() { return new
* SimpleAsyncTaskExecutor(); }
*/
@Bean
@JobScope
public Partitioner partitioner() {
MultiResourcePartitioner partitioner = new MultiResourcePartitioner();
Resource[] resources;
try {
resources = resourcePatternResolver.getResources("file:"
+ getFilesPath() + FILE_PATTERN);
} catch (IOException e) {
throw new RuntimeException(
"I/O problems when resolving the input file pattern.", e);
}
partitioner.setResources(resources);
return partitioner;
}
@Bean
@JobScope
public JobExecutionListener listener(ListItemWriter<T> writer) {
return new JobCompletionNotificationListener<T>(writer);
}
/*
* Use this if you want the writer to have job scope (JIRA BATCH-2269). Also
* change the return type of writer to ListItemWriter for this to work.
*/
@Bean
public TaskExecutor taskExecutor() {
return new SimpleAsyncTaskExecutor() {
@Override
protected void doExecute(final Runnable task) {
// gets the jobExecution of the configuration thread
final JobExecution jobExecution = JobSynchronizationManager
.getContext().getJobExecution();
super.doExecute(new Runnable() {
public void run() {
JobSynchronizationManager.register(jobExecution);
try {
task.run();
} finally {
JobSynchronizationManager.close();
}
}
});
}
};
}
}
假设为了讨论,我必须阅读发票数据。因此,我可以扩展上述类来创建InvoiceLoader:
@Configuration
public class InvoiceLoader extends AbstractFileLoader<Invoice>{
private class InvoiceFieldSetMapper implements FieldSetMapper<Invoice> {
public Invoice mapFieldSet(FieldSet f) {
Invoice invoice = new Invoice();
invoice.setNo(f.readString("INVOICE_NO");
return e;
}
}
@Override
public FieldSetMapper<Invoice> getFieldSetMapper() {
return new InvoiceFieldSetMapper();
}
@Override
public String getFilesPath() {
return "I:/CK/invoices/partitions/";
}
@Override
public String[] getColumnNames() {
return new String[] { "INVOICE_NO", "DATE"};
}
@Override
@Bean(name="invoiceJob")
public Job loaderJob(Step s1,
JobExecutionListener listener) {
return createJob(s1, listener);
}
@Override
public int getChunkSize() {
return 25254;
}
@Override
public int getThrottleLimit() {
return 8;
}
}
假设我还有一个名为 Inventory 的类扩展了 AbstractFileLoader.
在应用启动时,我可以如下加载这两个注解配置:
AbstractApplicationContext context1 = new AnnotationConfigApplicationContext(InvoiceLoader.class, InventoryLoader.class);
在我的应用程序的其他地方,两个不同的线程可以按如下方式启动作业:
线程 1:
JobLauncher jobLauncher1 = context1.getBean(JobLauncher.class);
Job job1 = context1.getBean("invoiceJob", Job.class);
JobExecution jobExecution = jobLauncher1.run(job1, jobParams1);
线程 2:
JobLauncher jobLauncher1 = context1.getBean(JobLauncher.class);
Job job1 = context1.getBean("inventoryJob", Job.class);
JobExecution jobExecution = jobLauncher1.run(job1, jobParams1);
这种方法的优点是每次有新文件要读取时,开发人员/用户所要做的就是继承AbstractFileLoader 并实现所需的抽象方法,而无需详细了解如何组装工作。
问题:
- 我是 Spring 批处理的新手,所以我可能忽略了这种方法的一些不太明显的问题,例如 Spring 批处理中的共享内部对象可能导致两个作业一起运行失败或明显的问题,例如豆子。
- 有没有更好的方法来实现我的目标?
-
@Value("#{stepExecutionContext['fileName']}")的fileName属性始终被赋值为I:/CK/invoices/partitions/,这是InvoiceLoader中getPath方法返回的值,即使getPathmethod inInventoryLoader`返回不同的值.
【问题讨论】:
-
这里的问题在哪里?
-
对不起,如果我不清楚。问题在最后两段。我想在旅途中组装一个新的
Job,而不是我启动一个预配置的工作开发人员将提供FieldSetMapper和ItemWriter的子类以及其他参数,例如提交间隔FileLoader。然后FileLoader应该将这些组件组装成Job并启动Job。我想代表开发人员组装作业,而不是要求他们配置它们。我的帖子中的第 1 点和第 2 点指定了要求。编辑我的帖子以使其清晰! -
那么您希望“用户”为
FieldSetMapper、ItemWriter等提供bean? -
@Artefacto 没错。我希望用户只提供
FieldSetMapper和ItemWriter以及我的帖子中提到的其他参数。我将使用这些组件并组装一个Job并启动它。所以这里没有预先配置的作业。我必须从提供的组件中组装一个,然后启动它。此外,此方法可能会在应用程序的生命周期内被多次调用,因此每个作业都必须是唯一的,但我应该能够重用诸如JobLauncher之类的组件。坦率地说,我不知道如何实现这一目标。 -
@Artefacto 组装作业而不是要求“用户”配置它们的原因是,“用户”主要是支持人员,他们知道一点点 Java 代码能够编写简单的 Java 类,如
FieldSetMapper实现。要求他们配置Spring batch工作是不可能的。
标签: spring file annotations spring-batch