Конфигурации диспетчера транзакций Hibernate в Spring - PullRequest
11 голосов
/ 27 февраля 2011

В моем проекте я использую Hibernate с программным разграничением транзакций. Каждый раз в моих методах обслуживания я пишу что-то похожее на это.

Session session = HibernateUtil.getSessionFactory().openSession();
session.beginTransaction();
.. perform operation here
session.getTransaction().commit();

Теперь я собираюсь реорганизовать свой код с помощью декларативного управления транзакциями. Что я получил сейчас ...

  <context:component-scan base-package="package.*"/>

  <bean id="mySessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
    <property name="configLocation" value="classpath:hibernate.cfg.xml"></property>
    <property name="configurationClass">
        <value>org.hibernate.cfg.AnnotationConfiguration</value>
    </property>
  </bean>

  <tx:annotation-driven transaction-manager="txManager"/>


  <bean id="txManager"  class="org.springframework.orm.hibernate3.HibernateTransactionManager">
    <property name="sessionFactory">
         <ref local="mySessionFactory"/>
    </property>
  </bean>

Класс обслуживания:

@Service
public class TransactionalService {

    @Autowired
    private SessionFactory factory;

    @Transactional
    public User performSimpleOperation() {
        return (User)factory.getCurrentSession().load(User.class, 1L);
    }
}

И простой тест -

@Test
public void useDeclarativeDem() {
    FileSystemXmlApplicationContext ctx = new FileSystemXmlApplicationContext("spring-config.xml");
    TransactionalService b = (TransactionalService)ctx.getBean("transactionalService");
     User op = b.performSimpleOperation();
     op.getEmail();

Когда я пытаюсь получить электронную почту пользователя вне Транзакционного метода, я получаю исключение инициализации Lazy, электронная почта - мой случай, является простой строкой. Hibernate даже не выполняет sql-запрос, пока я не вызову какие-либо методы получения на моем POJO.

1) что я здесь не так делаю?

2) Является ли этот подход действительным?

3) Можете ли вы предложить какой-либо проект с открытым исходным кодом, который работает над Spring / Hibernate с конфигурацией на основе аннотаций?

Обновление

По какой-то причине, если я заменю getCurrentSession на openSession , этот код работает нормально. Может кто-нибудь объяснить это, пожалуйста?

Спасибо

Ответы [ 3 ]

11 голосов
/ 28 февраля 2011

Хорошо, наконец я понял, в чем проблема. В коде выше я использовал load вместо get. Session.load на самом деле не попал в базу данных. Вот почему я получаю исключение отложенной инициализации вне метода @Transactional

Если я использую openSession вместо getCurrentSession, сессия открывается вне контейнера Spring. В результате сессия не была закрыта, и это позволило мне прочитать свойства объекта вне метода @Transactional

2 голосов
/ 27 февраля 2011

Причина, по которой Hibernate не выполняет никаких SQL-запросов до тех пор, пока вы не вызовете геттеры, заключается в том, что я считаю, что для FetchType установлено значение LAZY.Чтобы решить эту проблему, вам нужно изменить FetchType на EAGER в вашем POJO:

@Entity
@Table(name = "user")
public class User {

    /*Other data members*/

    @Basic(fetch = FetchType.EAGER)
    private String email;

}

Лично мне никогда не приходилось указывать базовые типы, чтобы иметь EAGER FetchType, поэтому я не совсем уверен, почему ваша конфигурация требуетэтот.Если это только в ваших тестах, это может быть связано с тем, как вы настроили свои тесты JUnit.В объявлении класса должно быть что-то вроде этого:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration({"/test-app-config.xml"})
@Transactional
public class UserServiceTest {

}

Что касается хорошего ресурса, я всегда нахожу SpringByExample полезным.

EDIT

Так что я не совсем уверен, что не так с вашей конфигурацией.Он отличается от того, как я настроил мой, так что вот моя типичная конфигурация в надежде, что это поможет.hibernate.transaction.factory_class может быть ключевым свойством, которое вам не хватает.Я также использую AnnotationSessionFactoryBean:

<!-- DataSource -->
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close"
    p:driverClassName="com.mysql.jdbc.Driver" 
    p:url="jdbc:mysql://localhost/dbname"
    p:username="root"
    p:password="password"/>

<!-- Hibernate session factory -->
<bean id="sessionFactory"
    class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean"
    p:dataSource-ref="dataSource">
    <property name="packagesToScan" value="com.beans"/>
    <property name="hibernateProperties">
        <props>
            <prop key="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</prop>
            <prop key="hibernate.show_sql">true</prop>
            <prop key="hibernate.transaction.factory_class">org.hibernate.transaction.JDBCTransactionFactory</prop>
        </props>
    </property>
</bean> 

<!-- Transaction Manager -->
<bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
    <property name="sessionFactory">
        <ref bean="sessionFactory" />
    </property>
</bean>

<tx:annotation-driven transaction-manager="transactionManager"/>
0 голосов
/ 13 февраля 2015

Из документации Hibernate https://docs.jboss.org/hibernate/orm/4.2/manual/en-US/html/ch11.html#objectstate-loading:

Имейте в виду, что load () сгенерирует неисправимое исключение, если нет соответствующей строки в базе данных. Если класс сопоставлен с прокси, load () просто возвращает неинициализированный прокси и фактически не обращается к базе данных, пока вы не вызовете метод прокси. Это полезно, если вы хотите создать ассоциацию с объектом без фактической загрузки его из базы данных. Он также позволяет загружать несколько экземпляров в виде пакета, если для отображения класса определен размер пакета.

Из приведенной выше документации в вашем случае bean-компонент сопоставлен с пружинным прокси, поэтому загрузка только возвращается. Следовательно, вам нужно использовать get () вместо load (), которая всегда попадает в базу данных.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...