java - 无论如何,我无法在 Hibernate 中批处理 MySQL INSERT 语句

标签 java mysql hibernate jdbc batch-processing

我目前正面临众所周知且常见的 Hibernate 插入批处理问题。

我需要保存 500 万行长的批处理。我首先尝试使用更轻的有效负载。由于我必须插入只有 2 种类型的实体(首先是 A 类型的所有记录,然后是 B 类型的所有记录,都指向公共(public)类型 C ManyToOne parent),我想充分利用来自 JDBC 批量插入。

我已经阅读了很多文档,但我尝试过的都不起作用。

  • 我知道,为了使用批量插入,我不能使用实体生成器。因此,我删除了 AUTO_INCREMENT ID,并使用技巧设置 ID:SELECT MAX(ID) FROM ENTITIES 并每次递增。
  • 我知道我必须定期刷新 session 。我会提前发布代码,但无论如何我每 500 个元素执行一次事务。
  • 我知道我必须将 hibernate.jdbc.batch_size 设置为与我的应用程序的批量大小一致,因此我将其设置在 LocalSessionFactoryBean 中(Spring ORM 集成)
  • 我知道我必须在连接 URL 中启用重写批处理语句。

这是我的实体

共同的父实体。这首先被插入到单个事务中。我不关心此处 的自动递增列。每个批处理作业只有一个记录

@Entity
@Table(...)
@SequenceGenerator(...)
public class Deal
{

    @Id
    @Column(
            name = "DEAL_ID",
            nullable = false)
    @GeneratedValue(
            strategy = GenerationType.AUTO)
    protected Long id;

    ................
}

其中一个 child (假设每批有 250 万条记录)

@Entity
@Table(
        name = "TA_LOANS")
public class Loan
{

    @Id
    @Column(
            name = "LOAN_ID",
            nullable = false)
    protected Long id;

    @ManyToOne(
            optional = false,
            targetEntity = Deal.class,
            fetch = FetchType.LAZY)
    @JoinColumn(
            name = "DEAL_ID",
            nullable = false)
    protected Deal deal;


    .............
}

其他 child 打字。假设其他250万条记录

@Entity
@Table(
        name = "TA_BONDS")
public class Bond
{

    @Id
    @Column(
            name = "BOND_ID")

    @ManyToOne(
            fetch = FetchType.LAZY,
            optional = false,
            targetEntity = Deal.class)
    @JoinColumn(
            name = "DEAL_ID",
            nullable = false,
            updatable = false)
    protected Deal deal;

}

插入记录的简化代码

    long loanIdCounter = loanDao.getMaxId(), bondIdCounter = bondDao.getMaxId(); //Perform SELECT MAX(ID)

    Deal deal = null;

    List<Bond> bondList = new ArrayList<Bond>(COMMIT_BATCH_SIZE); //500 constant value
    List<Loan> loanList = new ArrayList<Loan>(COMMIT_BATCH_SIZE);

    for (String msg: inputStreamReader)
    {
        log.debug(msg.toString());

        if (this is a deal)
        {
            Deal deal = parseDeal(msg.getMessage());

            deal = dealManager.persist(holder.deal); //Called in a separate transaction using Spring annotation @Transaction(REQUIRES_NEW)

        }
        else if (this is a loan)
        {

            Loan loan = parseLoan(msg.getMessage());
            loan.setId(++loanIdCounter);
            loan.setDeal(deal);

            loanList.add(loan);

            if (loanList.size() == COMMIT_BATCH_SIZE)
            {
                loanManager.bulkInsert(loanList); //Perform a bulk insert in a single transaction, not annotated but handled manually this time
                loanList.clear();
            }
        }
        else if (this is a bond)
        {
            Bond bond = parseBond(msg.getMessage());
            bond.setId(++bondIdCounter);
            bond.setDeal(deal);

            bondList.add(bond);



            if (bondList.size() == COMMIT_BATCH_SIZE) //As above
            {
                bondManager.bulkInsert(bondList);
                bondList.clear();

            }
        }
    }

    if (!bondList.isEmpty())
        bondManager.bulkInsert(bondList);
    if (!loanList.isEmpty())
        loanManager.bulkInsert(loanList);
    //Flush remaining items, not important

bulkInsert 的实现:

@Override
public void bulkInsert(Collection<Bond> bonds)
{
    // StatelessSession session = sessionFactory.openStatelessSession();
    Session session = sessionFactory.openSession();
    try
    {
        Transaction t = session.beginTransaction();
        try
        {
            for (Bond bond : bonds)
                // session.persist(bond);
                // session.insert(bond);
                session.save(bond);
        }
        catch (RuntimeException ex)
        {
            t.rollback();
        }
        finally
        {
            t.commit();
        }
    }
    finally
    {
        session.close();
    }

}

从评论中可以看出,我尝试了几种有状态/无状态session的组合。都没有用。

我的 dataSource 是一个 ComboPooledDataSource使用以下网址

<b:property name="jdbcUrl" value="jdbc:mysql://server:3306/db?autoReconnect=true&amp;rewriteBatchedStatements=true" />

我的SessionFactory

<b:bean id="sessionFactory" class="class.that.extends.org.springframework.orm.hibernate3.LocalSessionFactoryBean" lazy-init="false" depends-on="dataSource">
        <b:property name="dataSource" ref="phoenixDataSource" />
        <b:property name="hibernateProperties">
            <b:props>
                <b:prop key="hibernate.dialect">${hibernate.dialect}</b:prop> <!-- MySQL5InnoDb-->
                <b:prop key="hibernate.show_sql">${hibernate.showSQL}</b:prop>
                <b:prop key="hibernate.jdbc.batch_size">500</b:prop>
                <b:prop key="hibernate.jdbc.use_scrollable_resultset">false</b:prop>
                <b:prop key="hibernate.cache.use_second_level_cache">false</b:prop>
                <b:prop key="hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider</b:prop>
                <b:prop key="hibernate.cache.use_query_cache">false</b:prop>
                <b:prop key="hibernate.validator.apply_to_ddl">false</b:prop>
                <b:prop key="hibernate.validator.autoregister_listeners">false</b:prop>
                <b:prop key="hibernate.order_inserts">true</b:prop>
                <b:prop key="hibernate.order_updates">true</b:prop>
            </b:props>
        </b:property>
</b:bean>

即使我的项目范围的类扩展了 LocalSessionFactoryBean,它不会覆盖它的方法(只添加几个项目范围的方法)

几天来我很生气。我读了几篇文章,但没有一篇文章帮助我启用批量插入。我从使用 Spring 上下文检测的 JUnit 测试运行我的所有代码(因此我可以 @Autowire 我的类)。我所有的尝试都只产生了很多单独的 INSERT 语句

我错过了什么?

最佳答案

很可能您的查询正在被重写,但您不会通过查看 Hibernate SQL 日志知道是否如此。 Hibernate 不会重写插入语句——MySQL 驱动程序会重写它们。也就是说,Hibernate会向驱动发送多条insert语句,然后由驱动重写。所以 Hibernate 日志只显示 Hibernate 发送给驱动程序的 SQL,而不是驱动程序发送给数据库的 SQL。

您可以通过在连接 url 中启用 MySQL 的 profileSQL 参数来验证这一点:

<b:property name="jdbcUrl" value="jdbc:mysql://server:3306/db?autoReconnect=true&amp;rewriteBatchedStatements=true&amp;profileSQL=true" />

使用类似于您的示例,这就是我的输出:

insert into Person (firstName, lastName, id) values (?, ?, ?)
insert into Person (firstName, lastName, id) values (?, ?, ?)
insert into Person (firstName, lastName, id) values (?, ?, ?)
insert into Person (firstName, lastName, id) values (?, ?, ?)
insert into Person (firstName, lastName, id) values (?, ?, ?)
insert into Person (firstName, lastName, id) values (?, ?, ?)
insert into Person (firstName, lastName, id) values (?, ?, ?)
insert into Person (firstName, lastName, id) values (?, ?, ?)
insert into Person (firstName, lastName, id) values (?, ?, ?)
insert into Person (firstName, lastName, id) values (?, ?, ?)
Wed Feb 05 13:29:52 MST 2014 INFO: Profiler Event: [QUERY]  at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) duration: 1 ms, connection-id: 81, statement-id: 33, resultset-id: 0, message: insert into Person (firstName, lastName, id) values ('person1', 'Name', 1),('person2', 'Name', 2),('person3', 'Name', 3),('person4', 'Name', 4),('person5', 'Name', 5),('person6', 'Name', 6),('person7', 'Name', 7),('person8', 'Name', 8),('person9', 'Name', 9),('person10', 'Name', 10)

前 10 行由 Hibernate 记录,但这不是实际发送到 MySQL 数据库的内容。最后一行来自 MySQL 驱动程序,它清楚地显示了具有多个值的单个批插入,这就是实际发送到 MySQL 数据库的内容。

关于java - 无论如何,我无法在 Hibernate 中批处理 MySQL INSERT 语句,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47942757/

相关文章:

hibernate - JPA/Hibernate 中的复合键具有继承的类

mysql - hibernate可选关系不显示数据

java - Sqlite + java,准备好的语句有奇怪的行为

mysql - 查找日期范围内商店中存储架的可用性

hibernate - 在合并值中执行@Formula

javascript - 仅使用 Electron (Atom shell) 和 PHP+mysql 服务器的身份验证表单

Mysql针对每种类型的field2查找不在同一个表中的所有项目

用于 fiddler 文件的 Java API

java - Spring启动Jsp文件找不到

java - 从哪里获取 Ebay PaymentProfileID、ShippingProfileID 和 ReturnProfileID?