Skip to content
Advertisement

How to load the collection of a LAZY association for an already found entity

Consider the following:

JavaScript
JavaScript
JavaScript

The code above will throw org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: test.app.Car.wheels.

My question is how to implement loadWheelsForCar(Car c) method without having to find the car again.

With other words, how to SELECT * FROM WHEELS WHERE owner_car_id = car.id and add the result to the collection. I can probably do it manually, but is this the only way to go?

I am aware that LazyInitializationException is thrown when there is no active session(Doesn’t @Transactional cause the creation of a new one?). I have tried to:

JavaScript

but the exception is thrown.

In case of a XY problem, the reason I don’t do this (CarService):

JavaScript

is because the parent entity (Car entity) in the main application has multiple @OneToMany associations and some them have nested ones as well. If i follow this approach I will end up with multiple @Transactional methods like getCarByIdWithWheels, getCarByIdWithSeats, getCarByIdWithSeatsAndWheels, etc. But what I want is to be able to do something like:

JavaScript

I tried somethings found in web but every solution I found was “re-loading” the “Car” entity.

Advertisement

Answer

I am aware that LazyInitializationException is thrown when there is no active session(Doesn’t @Transactional cause the creation of a new one?)

Hibernate does create a new session. But that session is fresh and it is not aware of your Car c as that car is not fetched, saved or updated in that new session because you fetched that car in a different session.

  • You know that your car you are passing is exactly same as the car in the database and there is no update in between. Unfortunately there is no API in hibernate to tell that here is a car exactly as it is in database, don’t check it with it database and just believe you. There is no api like session.attach(car).

  • So the closest you can do is session.merge(car). Hibernate will issue a select to check if the car you are passing and the car in the database are same, and since it is same, it will not issue an update. As part of that select wheels would have been loaded too.

JavaScript
  • You will see the above method does not issue any update but just one select query similar to below
JavaScript
User contributions licensed under: CC BY-SA
5 People found this is helpful
Advertisement