Spring Batch JpaItemWriter vs HibernateItemWriter 以及为什么在使用 HibernateItemWriter 时需要 HibernateTransactionManager

Ale*_*Zai 6 java spring hibernate spring-batch spring-boot

我正在使用 Spring Boot 项目处理 Spring Batch。我的问题是,当我如下使用 HibernateItemWriter 时,为什么需要来自 LocalSessionFactoryBean 的 HibernateTransactionManager 和 SessionFactory?

应用程序.java

import java.util.Properties;
import javax.sql.DataSource;

import org.hibernate.SessionFactory;
import org.springframework.batch.core.configuration.annotation.EnableBatchProcessing;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.Bean;
import org.springframework.orm.hibernate5.HibernateTransactionManager;
import org.springframework.orm.hibernate5.LocalSessionFactoryBuilder;
import org.springframework.transaction.PlatformTransactionManager;

@SpringBootApplication
@EnableBatchProcessing
public class Application {

public static void main(String[] args) throws Exception {
    SpringApplication.run(Application.class, args);
}

@Bean
public PlatformTransactionManager transactionManager() {
    HibernateTransactionManager transactionManager = new HibernateTransactionManager();
    transactionManager.setSessionFactory(sessionFactory(null));
    return transactionManager;
}

@Bean
public SessionFactory sessionFactory(DataSource datasource) {
    Properties properties = new Properties();
    properties.setProperty("hibernate.show_sql", "true");
    properties.setProperty("hibernate.dialect", "org.hibernate.dialect.HSQLDialect");

    return new LocalSessionFactoryBuilder(datasource).scanPackages("hello")
             .addProperties(properties)
            .buildSessionFactory();
}
}
Run Code Online (Sandbox Code Playgroud)

批处理配置文件

import org.hibernate.SessionFactory;
import org.springframework.batch.core.Step;
import org.springframework.batch.core.configuration.annotation.JobBuilderFactory;
import org.springframework.batch.core.configuration.annotation.StepBuilderFactory;
import org.springframework.batch.core.configuration.annotation.StepScope;
import org.springframework.batch.item.ItemWriter;
import org.springframework.batch.item.database.HibernateItemWriter;
import org.springframework.batch.item.file.FlatFileItemWriter;
import org.springframework.batch.item.file.transform.BeanWrapperFieldExtractor;
import 
org.springframework.batch.item.file.transform.DelimitedLineAggregator;
import org.springframework.batch.item.file.transform.FieldExtractor;
import org.springframework.batch.item.file.transform.LineAggregator;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.FileSystemResource;
import org.springframework.transaction.PlatformTransactionManager;
import org.springframework.web.client.RestTemplate;

@Configuration
public class BatchConfiguration {

@Autowired
public JobBuilderFactory jobBuilderFactory;

@Autowired
public StepBuilderFactory stepBuilderFactory;

@Autowired
public RestTemplate restTemplate;


@Autowired
public PlatformTransactionManager tManager;

@Value("${file.name}")
public String fileName;


@Bean
@StepScope
public RestItemReader reader2() {
    return new RestItemReader(restTemplate);
}

@Bean
public PersonItemProcessor processor() {
    return new PersonItemProcessor();
}

@Bean
public HibernateItemWriter<Person> hibernateWriter(SessionFactory emf) {
    HibernateItemWriter<Person> writer = new HibernateItemWriter<>();
    writer.setSessionFactory(emf);
    return writer;
}


@Bean
public Step step1() {
    return stepBuilderFactory.get("step1")
            .transactionManager(tManager)
        .<KarvyFundInfoModel, Person> chunk(2)
        .reader(reader2())
        .processor(new PersonItemProcessor())
        .writer(hibernateWriter(null))
        .build();
}
}
Run Code Online (Sandbox Code Playgroud)

这是因为如果我不包含它,并且通过使用如下代码从 EntityManagerFactory 获取 SessionFactory

EntityManagerFactory.unwarp(SessionFactory.class);
Run Code Online (Sandbox Code Playgroud)

我会收到“没有正在进行的交易”错误。但是,当我使用 JpaItemWriter 时,情况并非如此。

但是根据我对 Spring Batch 的理解,在块处理中,已经提供了一个默认的事务管理器。

是否必须从 LocalSessionFactoryBean(来自 hibernate)提供 HibernateTransactionManager 和 SessionFactory 才能使用 HibernateItemWriter?

而且,JpaItemWriter 和 HibernateItemWriter 之间的主要区别是什么?我已经对这两个进行了研究,Jpa 是关于如何使用注解方式指定实体等的规范,而 hibernate 是 Jpa 的实现之一。不过,我对此不是很清楚。休眠是否比默认的 jpa 具有更多功能?例如 SearchCriteria 等?

Mah*_*ine 5

为什么在使用 HibernateItemWriter 时需要来自 LocalSessionFactoryBean 的 HibernateTransactionManager 和 SessionFactory

默认情况下,如果您提供DataSourcebean,Spring Batch 将使用 aDataSourceTransactionManager来管理事务。这个事务管理器对您的 JPA/Hibernate 上下文一无所知。因此,HibernateItemWriterSession幕后使用 Hibernate 的那个不“知道”由DataSourceTransactionManager. 因此错误:no transaction is in progress

HibernateTransactionManager就是使 HibernateSession参与 Spring 管理事务的原因。

JpaItemWriter 和 HibernateItemWriter 之间的主要区别是什么?

JpaItemWriter使用JPA的API(EntityManagerFactoryEntityManager)写的项目。它不使用任何 JPA 提供程序特定的 API。这使得可以在不更改编写器的情况下切换 JPA 提供程序。

HibernateItemWriter在另一侧使用Hibernate特定API(SessionFactorySession)和特定于仅休眠。此组件对于直接使用休眠而不使用 JPA 的应用程序非常有用。您可以使用相同的编写器,但对于另一个 JPA 提供程序,例如OpenJpaItemWriterEclipseLinkItemWriter使用来自这些提供程序的特定 API。


注意:有一个类似的问题,我在这里添加以供参考:Transaction management with Spring Batch