Spring @Transactional既作为动态Jdk代理又作为aspectj方面应用

时间:2010-08-26 08:10:21

标签: java spring aspectj spring-aop

我正在通过@Transactional注释向现有Java项目添加Spring声明性事务。

当我遇到问题(与此问题无关)时,我打开了完整的调试日志记录。奇怪的是,我注意到以下几点:

17:47:27,834 DEBUG HibernateTransactionManager:437 - Found thread-bound Session [org.hibernate.impl.SessionImpl@10ed8a8e] for Hibernate transaction
17:47:27,845 DEBUG HibernateTransactionManager:470 - Participating in existing transaction
17:47:27,865 DEBUG AnnotationTransactionAttributeSource:106 - Adding transactional method 'updateUserProfile' with attribute: PROPAGATION_REQUIRED,ISOLATION_DEFAULT; ''
17:47:27,875 DEBUG AnnotationTransactionAspect:321 - Skipping transactional joinpoint [se.myservice.UserService.updateUserProfile] because no transaction manager has been configured

经过一些调试后,我发现前三个日志条目,即它发现一个线程绑定会话并正在使用该事务,由我的UserService类上的JdkDynamicAopProxy生成。

最后一条日志消息看起来很惊人。它在方法执行之前在连接点调用。查看AnnotationTransactionAspect的源时,如果未设置事务管理器,则会生成此消息。在这种情况下,因为Spring从不在这方面执行任何依赖注入。

在我看来,两种不同的“风格”交易都适用:动态代理和方面。我唯一与交易相关的配置是:

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

我们在项目中使用了AspectJ,但是在我的aop.xml中没有注册AnnotationTransactionAspect方面。我们使用的是Spring 3.0.2.RELEASE。

我应该对此感到震惊吗? Spring会为我注册这方面吗?使用AspectJ时我不应该使用annotation-driven吗?

2 个答案:

答案 0 :(得分:8)

奇怪的是,听起来你有这样的配置:

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

(使用AspectJ的事务支持,而不是JDK代理)

由于您的配置没有mode属性,因此默认情况应该启动(代理模式)。但是AnnotationTransactionAspect是aspectj模式使用的确切方面。

答案 1 :(得分:2)

使用java config获取aspectj事务。

@EnableWebMvc
@Configuration
@ComponentScan("com.yourdomain")
@EnableTransactionManagement(mode=AdviceMode.ASPECTJ)
public class ApplicationConfig extends WebMvcConfigurerAdapter {

    @Bean
    public LocalContainerEntityManagerFactoryBean entityManagerFactory() {

        //...
    }

    @Bean
    public JpaTransactionManager transactionManager() {

        JpaTransactionManager bean = new JpaTransactionManager(entityManagerFactory().getObject());
        return bean ;
    }

    @Bean
    public AnnotationTransactionAspect annotationTransactionAspect() {

        AnnotationTransactionAspect bean = AnnotationTransactionAspect.aspectOf();
        bean.setTransactionManager(transactionManager());
        return bean;
    }
}

如果您使用的是maven:

<plugin>
    <groupId>org.codehaus.mojo</groupId>
    <artifactId>aspectj-maven-plugin</artifactId>
    <version>1.7</version>
    <configuration>
        <aspectLibraries>
            <aspectLibrary>
                <groupId>org.springframework</groupId>
                <artifactId>spring-aspects</artifactId>
            </aspectLibrary>
        </aspectLibraries>
        <complianceLevel>1.8</complianceLevel>
        <source>1.8</source>
        <target>1.8</target>
        <showWeaveInfo>true</showWeaveInfo>
    </configuration>
    <executions>
        <execution>
            <goals>
                <goal>compile</goal>
            </goals>
        </execution>
    </executions>
</plugin>

如果你正在使用eclipse,这将确保在eclipse中部署时完成编织:

http://www.eclipse.org/ajdt/