2013-05-15 4 views
1

Мое понимание @Transactional должно применяться только к методам обслуживания, которые должны возникать в транзакции (например, сеттеры). Скажем, у меня есть следующие два класса (DAO слой и обслуживание слоя соответственно) ...Весна требует @Transactional быть на услугах геттера?

@Service("playerService") 
public class PlayerServiceImpl implements PlayerService { 
    @Autowired 
    private PlayerDao playerDao; 

    @Override 
    public List<Player> getAll() { 
     return playerDao.getAll(); 
    } 


    @Override 
    @Transactional 
    public void addAllPlayers(final List<Player> players) { 
     playerDao.addAllPlayers(players); 
    } 
} 

@Repository("playerDao") 
public class PlayerDaoImpl implements PlayerDao { 

    @Autowired 
    private SessionFactory sessionFactory; 

    @SuppressWarnings("unchecked") 
    @Override 
    public List<Player> getAll() { 
     return (List<Player>) sessionFactory.getCurrentSession() 
       .createQuery("FROM Player").list(); 
    } 
    @Override 
    public void addPlayer(final Player player) { 
     sessionFactory.getCurrentSession().save(player); 
    } 
} 

Теперь, если я назвал addAllPlayers() это работает отлично, никаких проблем вообще. Но когда я использую getAll(), sessionFactory.getCurrentSession выдает исключение HibernateException, сеанс не найден для текущего потока.

Если я добавлю @Transactional в сервисный уровень для getAll(), теперь это будет работать «отлично». Проблема с этим заключается в том, что мне не нужно открывать транзакцию только для вызова получателя.

Может ли кто-нибудь подумать о какой-либо причине, почему мне нужно добавить @Transactional на метод getter, чтобы получить sessionFactory для текущей сессии? Мой сервлет-context.xml и настойчивость-context.xml показаны ниже (они обе ссылки в моей web.xml в contextConfigLocation)

сервлет-context.xml

<?xml version="1.0" encoding="UTF-8"?> 
<beans:beans xmlns:mvc="http://www.springframework.org/schema/mvc" 
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:beans="http://www.springframework.org/schema/beans" 
    xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context" 
    xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd 
     http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd 
     http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd 
     http://www.springframework.org/schema/tx 
     http://www.springframework.org/schema/tx/spring-tx-3.0.xsd"> 

    <!-- DispatcherServlet Context: defines this servlet's request-processing 
     infrastructure --> 

    <!-- Enables the Spring MVC @Controller programming model --> 
    <tx:annotation-driven transaction-manager="hibernateTransactionManager" /> 

    <mvc:annotation-driven /> 
    <!-- Handles HTTP GET requests for /resources/** by efficiently serving 
     up static resources in the ${webappRoot}/resources directory --> 
    <mvc:resources location="/resources/css/" mapping="/css/**" /> 
    <mvc:resources location="/resources/js/" mapping="/js/**" /> 
    <mvc:resources location="/resources/images/" mapping="/images/**" /> 
    <mvc:resources location="/resources/img/" mapping="/img/**" /> 
    <mvc:resources location="/favicon.ico" mapping="/favicon.ico" /> 
    <!-- Resolves views selected for rendering by @Controllers to .jsp resources 
     in the /WEB-INF/views directory --> 
    <beans:bean 
     class="org.springframework.web.servlet.view.InternalResourceViewResolver"> 
     <beans:property name="prefix" value="/WEB-INF/views/" /> 
     <beans:property name="suffix" value=".jsp" /> 
    </beans:bean> 



    <context:component-scan base-package="com.footieview.app" /> 
    <beans:bean 
     class="org.springframework.web.servlet.view.ContentNegotiatingViewResolver"> 
     <beans:property name="mediaTypes"> 
      <beans:map> 
       <beans:entry key="html" value="text/html" /> 
       <beans:entry key="json" value="application/json" /> 
      </beans:map> 
     </beans:property> 
     <beans:property name="defaultViews"> 
      <beans:list> 
       <beans:bean 
        class="org.springframework.web.servlet.view.json.MappingJacksonJsonView"> 
        <beans:property name="prefixJson" value="true" /> 
       </beans:bean> 
      </beans:list> 
     </beans:property> 
    </beans:bean> 
    <beans:bean id="PlayerImportDaoImpl" 
     class="com.footieview.app.importer.dao.PlayerImportDaoImpl" /> 
    <beans:bean id="hibernateTransactionManager" 
     class="org.springframework.orm.hibernate4.HibernateTransactionManager"> 
     <beans:property name="sessionFactory" ref="sessionFactory" /> 
    </beans:bean> 

</beans:beans> 

-context.xml настойчивость

<?xml version="1.0" encoding="UTF-8"?> 
<beans:beans xmlns:mvc="http://www.springframework.org/schema/mvc" 
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:beans="http://www.springframework.org/schema/beans" 
    xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context" 
    xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd 
     http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd 
     http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd 
     http://www.springframework.org/schema/tx 
     http://www.springframework.org/schema/tx/spring-tx-3.0.xsd"> 
     <beans:bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" 
     destroy-method="close"> 
     <beans:property name="driverClassName" value="com.mysql.jdbc.Driver" /> 
     <beans:property name="url" 
      value="jdbc:mysql://localhost/db" /> 
     <beans:property name="username" value="username" /> 
     <beans:property name="password" value="password" /> 
    </beans:bean> 

    <beans:bean id="sessionFactory" 
     class="org.springframework.orm.hibernate4.LocalSessionFactoryBean"> 
     <beans:property name="annotatedClasses"> 
      <beans:list> 
       <beans:value>com.footieview.app.entity.Player</beans:value> 
      </beans:list> 
     </beans:property> 
     <beans:property name="dataSource" ref="dataSource" /> 
     <beans:property name="packagesToScan" value="com.footieview.app.entity.*" /> 
     <beans:property name="hibernateProperties"> 
      <beans:props> 
       <beans:prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect 
       </beans:prop> 
       <beans:prop key="hibernate.transaction.factory_class">org.hibernate.transaction.JDBCTransactionFactory 
       </beans:prop> 
       <beans:prop key="hibernate.show_sql">false</beans:prop> 
       <beans:prop key="hibernate.hbm2ddl.auto">create</beans:prop> 
       <beans:prop key="hibernate.cache.use_second_level_cache"> 
        true 
       </beans:prop> 
       <beans:prop key="hibernate.cache.provider_class"> 
        org.hibernate.cache.EhCacheProvider 
       </beans:prop> 
       <beans:prop key="hibernate.cache.use_query_cache"> 
        true 
       </beans:prop> 
       <beans:prop key="hibernate.cache.region.factory_class"> 
        org.hibernate.cache.ehcache.EhCacheRegionFactory 
       </beans:prop> 
       <beans:prop key="hibernate.cglib.use_reflection_optimizer"> 
        true 
       </beans:prop> 
      </beans:props> 
     </beans:property> 
    </beans:bean> 
</beans:beans> 

ответ

4

@Transactional не только открывает транзакцию DB, но и в случае Hibernate также создает сеанс спящего режима, если его нет. Типичный подход заключается в использовании OpenSessionInViewFilter, который создает один сеанс Hibernate для каждого HTTP-запроса.

Если вы не хотите использовать этот фильтр, вам также необходимо аннотировать геттеры с @Transactional.

+0

Вы можете использовать атрибут readOnly, чтобы уменьшить влияние производительности, если оно есть - @Transactional (readonly = true). – gkamal

+0

Правильно ли я считаю, что @Transactional создает эффект производительности, поскольку я понял, что если транзакция была открыта, она заблокировала бы таблицу за период транзакции. – david99world

+0

Транзакция не блокирует таблицу. Только модификации. То, что заблокировано, зависит от db - для блокировок оракулов используется ряд строк, для hsqldb это таблица по умолчанию, я думаю. – mrembisz

Смежные вопросы