Grails, вставка большого количества данных с использованием результатов withTransaction в OutOfMemoryError

java hibernate spring grails transactions

15326 просмотра

2 ответа

Я использую Grails 1.1 beta2. Мне нужно импортировать большой объем данных в мое приложение Grails. Если я неоднократно создаю экземпляр класса домена Grails, а затем сохраняю его, производительность будет недопустимо низкой. Возьмем, к примеру, импорт людей из телефонной книги:

for (each person in legacy phone book) {
    // Construct new Grails domain class from legacy phone book person
    Person person = new Person(...)
    person.save()
}

Это оказывается мучительно медленным. Кто-то из списка рассылки Grails предлагает пакетное сохранение в транзакции. Итак, теперь у меня есть:

List batch = new ArrayList()
for (each person in legacy phone book) {
    // Construct new Grails domain class from legacy phone book person
    Person person = new Person(...)
    batch.add(person)
    if (batch.size() > 500) {
        Person.withTransaction {
            for (Person p: batch)
                p.save()
            batch.clear()
        }
    }
}
// Save any remaining
for (Person p: batch)
    p.save()

Это должно работать быстрее, по крайней мере, в начале. Каждая транзакция сохраняет 500 записей. Со временем транзакции занимают все больше и больше времени. Первые несколько транзакций занимают около 5 секунд, затем они просто ползут оттуда. После примерно 100 транзакций каждая занимает более минуты, что опять-таки недопустимо. Хуже всего то, что со временем у Grails закончится куча памяти Java. Я могу увеличить размер кучи JVM, но это только задерживает OutOfMemoryErrorисключение.

Есть идеи, почему это так? Как будто там какой-то внутренний ресурс не выпускается. Производительность ухудшается, память удерживается, а затем системе не хватает памяти.

Согласно документации Grails , withTransactionпередается замыкание на TransactionStatusобъект Spring . Я не мог найти ничего, TransactionStatusчтобы закрыть / завершить транзакцию.

Редактировать: я запускаю это из консоли Grails ( grails console)

Изменить: Вот исключение из нехватки памяти:

Exception thrown: Java heap space

java.lang.OutOfMemoryError: Java heap space
    at org.hibernate.util.IdentityMap.entryArray(IdentityMap.java:194)
    at org.hibernate.util.IdentityMap.concurrentEntries(IdentityMap.java:59)
    at org.hibernate.event.def.AbstractFlushingEventListener.prepareEntityFlushes(AbstractFlushingEventListener.java:113)
    at org.hibernate.event.def.AbstractFlushingEventListener.flushEverythingToExecutions(AbstractFlushingEventListener.java:65)
    at org.hibernate.event.def.DefaultFlushEventListener.onFlush(DefaultFlushEventListener.java:26)
    at org.hibernate.impl.SessionImpl.flush(SessionImpl.java:1000)
    at org.hibernate.impl.SessionImpl.managedFlush(SessionImpl.java:338)
    at org.hibernate.transaction.JDBCTransaction.commit(JDBCTransaction.java:106)
    at org.springframework.orm.hibernate3.HibernateTransactionManager.doCommit(HibernateTransactionManager.java:655)
    at org.springframework.transaction.support.AbstractPlatformTransactionManager.processCommit(AbstractPlatformTransactionManager.java:732)
    at org.springframework.transaction.support.AbstractPlatformTransactionManager.commit(AbstractPlatformTransactionManager.java:701)
    at org.springframework.transaction.support.TransactionTemplate.execute(TransactionTemplate.java:140)
Автор: Steve Kuo Источник Размещён: 02.07.2019 01:27

Ответы (2)


12 плюса

Решение

Это общая проблема для всех приложений гибернации, и она вызвана ростом сеанса гибернации. Я предполагаю, что консоль Grails поддерживает открытый сеанс гибернации аналогично шаблону «открыть сеанс в представлении», который, как я знаю, используется для обычных веб-запросов.

Решение состоит в том, чтобы получить текущий сеанс и очистить его после каждого пакета. Я не уверен, как вы получаете доступ к Spring Bean с помощью консоли, обычно для контроллеров или сервисов, вы просто объявляете их как участников. Тогда вы можете получить текущий сеанс с sessionFactory.getCurrentSession(). Для того, чтобы очистить его, просто позвоните session.clear(), или, если вы хотите, чтобы выборочно использовать session.evict(Object)для каждого Personобъекта.

для контроллера / сервиса:

class FooController {
    def sessionFactory

    def doStuff = {
        List batch = new ArrayList()
        for (each person in legacy phone book) {
            // Construct new Grails domain class from legacy phone book person
            Person person = new Person(...)
            batch.add(person)
            if (batch.size() > 500) {
                Person.withTransaction {
                    for (Person p: batch)
                        p.save()
                    batch.clear()
                }
                // clear session here.
                sessionFactory.getCurrentSession().clear();
            }
        }
        // Save any remaining
        for (Person p: batch)
            p.save()
        }
    }
}

Надеюсь это поможет.

Автор: Gareth Davis Размещён: 10.01.2009 09:45

15 плюса

Тед Нейлид написал отличную запись в блоге об улучшении производительности партии. В том числе здесь в качестве ссылки.

Автор: Jean Barmash Размещён: 28.10.2009 04:03
Вопросы из категории :
32x32