所以我的春季教育仍在继续。目前我正在尝试学习一些注释以及它们为Spring 3带来的东西。所以我有一个迷你webapp可以连接到DB并通过表单放入内容并显示记录等等。一切正常。我决定尝试让Spring自动检测我标记为@Transactional的服务bean,但这样做会阻止应用程序保存到数据库。所以:
@Transactional
public class ReservationServiceImpl implements ReservationService {
有效。我在springcourt-data.xml文件中有这个类的bean声明。没问题。当我这样做时:
@Transacational
@Service("reservationService")
public class ReservationServiceImpl implements ReservationService {
它不再有效。我确实有
<context:component-scan base-package="com.springcourt" />
springcourt-servlet.xml文件中的。那么有谁能告诉我我搞砸了什么?我所做的就是为这个类添加另一个注释,并从xml文件中删除bean定义,它不再将数据保存到数据库中。我仍然可以从数据库中查询记录和内容,但显然它正在使用自动检测的服务bean。
以下是配置文件:
springcourt-servlet.xml
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd
http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd">
<context:component-scan base-package="com.springcourt" />
<bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping" />
<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">
<property name="webBindingInitializer">
<bean class="com.springcourt.web.ReservationBindingInitializer" />
</property>
</bean>
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/"/>
<property name="suffix" value=".jsp"/>
</bean>
</beans>
和
springcourt-data.xml
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-2.0.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-2.0.xsd">
<bean class="org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor" />
<bean id="entityManagerFactory"
class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
<property name="dataSource" ref="dataSource" />
<property name="jpaVendorAdapter">
<bean
class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
<property name="database" value="MYSQL" />
<property name="showSql" value="true" />
</bean>
</property>
</bean>
<bean id="dataSource"
class="org.apache.commons.dbcp.BasicDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver" />
<property name="url" value="jdbc:mysql://localhost:3306/test" />
<property name="username" value="root" />
<property name="password" value="admin" />
<property name="initialSize" value="5" />
</bean>
<bean id="transactionManager"
class="org.springframework.orm.jpa.JpaTransactionManager">
<property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>
<tx:annotation-driven />
<bean id="reservationService" class="com.springcourt.service.ReservationServiceImpl"/>
</beans>
答案 0 :(得分:3)
使用@Service和组件扫描时,bean由调度程序servlet(mvc)创建的上下文创建。由于事务:注释驱动是在根应用程序上下文中定义的,因此它不适用于servlet上下文中的bean。您可以通过删除@Service并将bean定义移动到servlet上下文文件来验证这一点 - 您应该看到相同的结果。
当您不使用组件扫描时 - bean在根应用程序上下文的XML中定义。
修复方法是将Web图层中的组件扫描标记更改为仅包含Web图层类 - 通过使用不同的基础包或使用包含/排除过滤器。在根应用程序上下文中为其他bean添加另一个组件扫描。
查询可能有效,因为您可能配置了OpenEntityManagerInViewInterceptor / Filter。
答案 1 :(得分:1)
由于您可以通过同一个bean查询数据库,因此您的@Transactional
可以正常工作,或者您经常会遇到异常(至少使用Hibernate)。最有可能在保存操作中,您会遇到一些导致事务回滚的运行时异常。试着找出什么是异常并从那里开始。
更新
要查看是否已应用@Transactional
,请从方法内部打印堆栈跟踪。如果您看到具有大量事务拦截器的长堆栈跟踪,则意味着事务方面有效。
答案 2 :(得分:1)
我有同样的问题并解决它。
首先,您必须将上下文分开:组件扫描到Web和数据级别,如下所示:
<!--in springcourt-servlet.xml -->
<context:component-scan base-package="com.springcourt.web" />
<!--in springcourt-data.xml -->
<context:component-scan base-package="com.springcourt.dao" />
第二次添加到springcourt-data.xml
<aop:aspectj-autoproxy/>
我希望它会有所帮助
答案 3 :(得分:0)
试试这个
将上下文组件扫描添加到spring-court-data.xml
<context:component-scan base-package="com.springcourt" />
单独测试服务,创建一个像这样的JUNIT测试
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"classpath:springcourt-data.xml"})
@Transactional
public class ReservationServiceImplTest()
{
@Autowired
ReservationServiceImpl service;
@Test
public void validateContext()
{
Assert.assertNotNull(service);
}
@Test
@Rollback(false)
public void save()
{
service.save(data);
}
}
答案 4 :(得分:0)
一个非常好的替代方法是使用以下内容:`
@Configuration
@EnableAspectJAutoProxy(proxyTargetClass=true)
public class AppConfig {
(...)
}
您需要做的是将所有内容注入相同的范围。一种方法是更改ApplicationContext xml,如前所述,另一种方法是使用类似于Spring代理的CGLIB,以便获得基于子类的代理,并在那里编写bean实现和定义。
进一步阅读: