Grails, Inserting lots of data using withTransaction results in OutOfMemoryError

后端 未结 2 548
别跟我提以往
别跟我提以往 2021-02-07 23:35

I\'m using Grails 1.1 beta2. I need to import a large amount of data into my Grails application. If I repeatedly instantiate a grails domain class and then save it, the performa

2条回答
  •  一个人的身影
    2021-02-08 00:05

    This is a common issue with all hibernate applications and it is caused by the growth of the hibernate session. I'm guessing that the grails console holds a hibernate session open for you in a similar way to the 'open session in view' pattern that I know it uses in for normal web requests.

    The solution is to get hold of the current session and clear it after each batch. I'm not sure how you get hold of spring bean using the console, normally for controllers or services you just declare them as members. Then you can get the current session with sessionFactory.getCurrentSession(). In order to clear it just call session.clear(), or if you what to be selective use session.evict(Object) for each Person object.

    for a controller/service:

    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()
            }
        }
    }
    

    Hope this helps.

提交回复
热议问题