【问题标题】:Hibernate JDBC Batch size is not working休眠 JDBC 批量大小不起作用
【发布时间】:2014-02-12 01:34:01
【问题描述】:

我在 jpa 中使用 SpringFramework 3 和 Hibernate 4 和 MySQL 5。我的测试代码看起来像...

@Repository
public class TestRepositoryImpl implements TestRepository {

  @PersistenceContext
  private EntityManager em;

  @Override
  @Transactional
  public void insertBulk() {
     Item it;
     for(int i= 0; i<1000;i++) {
        it = new Item();
        it.setPrice(Math.random()*100);
        em.persist(it);
     }
  }
}

我的弹簧配置

 <bean id="entityManagerFactory"
    class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
    <property name="persistenceUnitName" value="application" />
</bean>

<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
    <property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>

<tx:annotation-driven transaction-manager="transactionManager" />

我的persistence.xml

<persistence xmlns="http://java.sun.com/xml/ns/persistence"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_1_0.xsd"
version="1.0">

<persistence-unit name="application" transaction-type="RESOURCE_LOCAL">
    <provider>org.hibernate.ejb.HibernatePersistence</provider>
    <class>com.springapp.test.domain.Item</class>
    <class>com.springapp.test.domain.Order</class>
    <exclude-unlisted-classes>true</exclude-unlisted-classes>
    <properties>
        <property name="hibernate.show_sql" value="true" />
        <property name="hibernate.format_sql" value="false" />
        <property name="hibernate.connection.driver_class" value="com.mysql.jdbc.Driver" />
        <property name="hibernate.connection.url" value="jdbc:mysql://localhost:3306/testdb" />
        <property name="hibernate.connection.username" value="root" />
        <property name="hibernate.connection.password" value="" />
        <property name="hibernate.dialect" value="org.hibernate.dialect.MySQL5Dialect" />
        <property name="hibernate.hbm2ddl.auto" value="update" />
        <property name="hibernate.jdbc.batch_size" value="20" />
    </properties>
</persistence-unit>

</persistence>

当我调用运行我的代码时,它将触发插入查询 1000 次,而不是触发 50 次插入查询。问题是什么? 请帮助我使用休眠在 jpa 中批量插入

【问题讨论】:

  • 请添加堆栈跟踪
  • 对不起...这段代码没有异常或任何错误。代码工作正常,但在控制台上打印 1000 次插入查询。

标签: java mysql spring hibernate jpa


【解决方案1】:

您误解了“批量大小”。批量大小意味着一次性发送“批量大小”数量的查询,而不是在代码触发查询时发送每个查询。因此在这种情况下,将有 1000 个插入查询,发送 50 次,每批 20 个插入查询。

【讨论】:

  • 我想做这样的link
【解决方案2】:

请注意,如果插入表的主键是GenerationType.Identity,Hibernate 将在 JDBC 级别透明地禁用插入批处理。

save() 只有一条记录然后flush(),所以每次flush 只需要处理一个附加的INSERT SQL。这就是为什么 Hibernate 不能帮助您批量插入,因为只有一个 INSERT SQL 需要处理。您应该在调用 flush() 之前 save() 达到一定数量的记录,而不是为每个 save() 调用 flush()。

批量插入的最佳实践是这样的:

Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
for  ( int i=0; i<888888; i++ ) {
  TableA record = new TableA();
    record.setXXXX();
    session.save(record)
    if ( i % 50 == 0 ) { //50, same as the JDBC batch size
        //flush a batch of inserts and release memory:
        session.flush();
        session.clear();
    }
}
tx.commit();
session.close();

您逐批保存和刷新记录。在每个批次结束时,您应该清除持久性上下文以释放一些内存,以防止内存耗尽,因为每个持久性对象都被放置到第一级缓存(您的 JVM 的内存)中。您还可以禁用二级缓存以减少不必要的开销。

请查看此链接http://docs.jboss.org/hibernate/orm/3.5/reference/en/html/batch.html

【讨论】:

【解决方案3】:

在级别调试中添加记录器org.hibernate.engine.jdbc.batch.internal.BatchingBatch。 在插入顺序错误的情况下,hibernate 可以生成大小为 1 或 2 的批次。 尝试使用hibernate.order_inserts=true hibernate.order_updates=true

【讨论】:

    猜你喜欢
    • 2011-07-15
    • 1970-01-01
    • 2018-03-14
    • 2021-10-02
    • 1970-01-01
    • 2021-04-12
    • 2020-09-26
    • 2017-12-10
    • 1970-01-01
    相关资源
    最近更新 更多