Spring, @Transactional and Hibernate Lazy Loading

前端 未结 3 1896
孤独总比滥情好
孤独总比滥情好 2021-02-04 10:16

i\'m using spring + hibernate. All my HibernateDAO use directly sessionFactory.

I have application layer -> service layer -> DAO layer and all collections is lazly loade

相关标签:
3条回答
  • 2021-02-04 10:45

    Hibernate recently introduced fetch profiles which (in addition to performance tuning) is ideal for solving issues like this. It allows you to (at runtime) choose between different loading and initialization strategies.

    http://docs.jboss.org/hibernate/core/3.5/reference/en/html/performance.html#performance-fetching-profiles

    Edit (added section on how to set the fetch profile using an interceptor):

    Before you get started: Check that fetch profiles actually will work for you. I haven't used them myself and see that they are currently limited to join fetches. Before you waste time on implementing and wiring up the interceptor, try setting the fetch profile manually and see that it actually solves your problem.

    There are many ways to setup interceptors in Spring (according to preference), but the most straight-forward way would be to implement a MethodInterceptor (see http://static.springsource.org/spring/docs/3.0.x/spring-framework-reference/html/aop-api.html#aop-api-advice-around). Let it have a setter for the fetch profile you want and setter for the Hibernate Session factory:

    public class FetchProfileInterceptor implements MethodInterceptor {
    
        private SessionFactory sessionFactory;
        private String fetchProfile;
    
        ... setters ...    
    
        public Object invoke(MethodInvocation invocation) throws Throwable {
            Session s = sessionFactory.openSession(); // The transaction interceptor has already opened the session, so this returns it.
            s.enableFetchProfile(fetchProfile);
            try {
                return invocation.proceed();
            } finally {
                s.disableFetchProfile(fetchProfile);
            }
        }
    }
    

    Lastly, enable the interceptor in the Spring config. This can be done in several ways and you probably already have a AOP setup that you can add it to. See http://static.springsource.org/spring/docs/3.0.x/spring-framework-reference/html/aop.html#aop-schema.

    If you're new to AOP, I'd suggest trying the "old" ProxyFactory way first (http://static.springsource.org/spring/docs/3.0.x/spring-framework-reference/html/aop-api.html#aop-api-proxying-intf) because it's easier to understand how it works. Here's some sample XML to get you started:

    <bean id="fetchProfileInterceptor" class="x.y.zFetchProfileInterceptor">
      <property name="sessionFactory" ref="sessionFactory"/>
      <property name="fetchProfile" ref="gui-profile"/>
    </bean>
    
    <bean id="businessService" class="x.y.x.BusinessServiceImpl">
      <property name="dao" .../>
      ...
    </bean>
    
    <bean id="serviceForSwinGUI" 
        class="org.springframework.aop.framework.ProxyFactoryBean">
        <property name="proxyInterfaces" value="x.y.z.BusinessServiceInterface/>
    
        <property name="target" ref="businessService"/>
        <property name="interceptorNames">
            <list>
                <value>existingTransactionInterceptorBeanName</value>
                <value>fetchProfileInterceptor</value>
            </list>
        </property>
    </bean>
    
    0 讨论(0)
  • 2021-02-04 11:00
    1. Create a method in the service layer that returns the lazy-loaded object for that entity
    2. Change to fetch eager :)
    3. If possible extend your transaction into the application layer

    (just while we wait for someone who knows what they are talking about)

    0 讨论(0)
  • 2021-02-04 11:07

    You need to rework your session management, unfortunately. This is a major problem when dealing with Hibernate and Spring, and it's a gigantic hassle.

    Essentially, what you need is for your application layer to create a new session when it gets your Hibernate object, and to manage that and close the session properly. This stuff is tricky, and non-trivial; one of the best ways to manage this is to mediate the sessions through a factory available from your application layer, but you still need to be able to end the session properly, so you have to be aware of the lifecycle needs of your data.

    This stuff is the most common complaint about using Spring and Hibernate in this way; really, the only way to manage it is to get a good handle on exactly what your data lifecycles are.

    0 讨论(0)
提交回复
热议问题