Анотацията Propagation.NEVER не работи

Програмисти! Не разбирам как работи атрибутът propagation в анотацията @Transactional. Моля помогнете)

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

<bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
    <property name="sessionFactory" ref="sessionFactory"/>
</bean>

<bean id="sessionFactory"
        class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
        <property name="packagesToScan" value="com.springapp.mvc"></property>
        <property name="hibernateProperties">
            <props>
                <prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
                <prop key="hibernate.show_sql">true</prop>
                <prop key="hibernate.jdbc.batch_size">10</prop>
                <prop key="hibernate.max_fetch_depth">3</prop>
                <prop key="jdbc.fetch_size">50</prop>
                <prop key="hbm2ddl.auto">create-drop</prop>
            </props>
        </property>
        <property name = "dataSource" ref = "dataSource"></property>
    </bean>

Клас на обслужване:

   @Service
   public class BookManager implements IBookManager {

      @Autowired
      private  SessionFactory  sessionFactory;

      @Override
      @Transactional(propagation = Propagation.REQUIRES_NEW)
      public void method1() {
          Session session = sessionFactory.getCurrentSession();
          Book book = (Book) session.load(Book.class, 1);
          System.out.println("first: " + book.getTitle());
         method2();
      }

      @Override
      @Transactional(propagation = Propagation.NEVER)
      public void method2() {
            System.out.println("hello");
     }
 }

Очаквам, че методът method2() хвърля изключение, тъй като е анотиран с Propagation.NEVER. Но това не се случва.

Изход

 июн 27, 2015 3:35:35 PM org.hibernate.dialect.Dialect <init>
 INFO: HHH000400: Using dialect: org.hibernate.dialect.MySQLDialect
 июн 27, 2015 3:35:35 PM  org.hibernate.engine.jdbc.internal.LobCreatorBuilder useContextualLobCreation
 INFO: HHH000424: Disabling contextual LOB creation as createClob() method threw error : java.lang.reflect.InvocationTargetException
 июн 27, 2015 3:35:36 PM org.hibernate.engine.transaction.internal.TransactionFactoryInitiator initiateService
 INFO: HHH000399: Using default transaction strategy (direct JDBC transactions)
 июн 27, 2015 3:35:36 PM org.hibernate.hql.internal.ast.ASTQueryTranslatorFactory <init>
 INFO: HHH000397: Using ASTQueryTranslatorFactory
 июн 27, 2015 3:35:36 PM org.springframework.orm.hibernate4.HibernateTransactionManager           afterPropertiesSet
 INFO: Using DataSource [org.apache.commons.dbcp.BasicDataSource@6c5945a7] of Hibernate SessionFactory for HibernateTransactionManager
 Hibernate: select book0_.id as id1_0_0_, book0_.title as title2_0_0_ from books book0_ where book0_.id=?
 first: 33
 hello

 Process finished with exit code 0

Защо не работи?

Благодаря :)


person Konstantin B.    schedule 27.06.2015    source източник
comment
Би било чудесно, ако можете да добавите решения за проследяване на стека на изключение и списък, които вече сте опитвали   -  person Viraj Nalawade    schedule 27.06.2015
comment
@VirajNalawade, виж какво се случва на конзолата   -  person Konstantin B.    schedule 27.06.2015


Отговори (1)


Транзакциите в Spring са базирани на прокси: когато bean A извиква транзакционен bean B, той всъщност извиква метод на динамичен прокси, който се занимава с отварянето на транзакцията, след това делегира към действителния bean B, след което се занимава с ангажирането /връщане на транзакцията.

Ако извикате method2 от method1 на един bean A, вашето извикване вече не се прихваща от транзакционния прокси и Spring напълно не знае, че method2() е извикан. Така че нищо не може да провери дали няма транзакция.

Поставете method2 в друг компонент, инжектиран в BookManager, и всичко ще работи както се очаква.

person JB Nizet    schedule 27.06.2015
comment
Благодаря! Вашият вариант наистина работи. Все пак прочетох документация, че още един начин за решение на моя проблем е следният: ‹tx:annotation-driven mode=aspectj/› В резултат на това получавам следната следа на стека: xception в основната нишка org.hibernate.HibernateException: Не може да се получи сесия, синхронизирана с транзакция за текущата нишка. Може би ще подскажеш какво правя неправилно? - person Konstantin B.; 27.06.2015
comment
Никаква идея. Никога не използвам този режим. - person JB Nizet; 27.06.2015