我正在尝试使用 Spring MVC 3 和 Hibernate 4.1 来获得声明式事务管理,为整个请求长度提供一个开放会话。我只有一个用于数据访问的“管理器”层,它直接使用会话。这是用@Transactional 注释的,并且会话在退出该层时关闭,并且我在控制器中得到一个延迟加载异常。
因此,我添加了 OpenSessionInViewFilter 过滤器,并且日志表明此过滤器已正确配置,并且它们还显示此过滤器正在打开会话。不幸的是,我的管理层仍然像以前一样打开和关闭自己的会话,并且我在控制器中得到了相同的延迟加载异常。
编辑:注意到在我的日志中,我打开了过滤器会话,打开了 HibernateTransactionManager 会话,关闭了 HibernateTransactionManager 会话,延迟加载异常,然后关闭了过滤器会话。所以我知道在延迟加载异常期间某处有一个打开的会话,但是该对象是在与另一个关闭的会话关联的事务中加载的。
我认为从管理器类中删除 @Transactional 会从该层删除会话管理并让过滤器完成它的工作,但是 sessionFactory().getCurrentSession() 只是获得一个关闭的会话;它无权访问过滤器提供的打开会话。
如何访问 OpenSessionInViewFilter 的明确打开的会话?
springapp-servlet.xml
<beans ..
<context:component-scan base-package="springapp.web" />
<bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"/>
<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter"/>
<!-- Enables the Spring MVC @Controller programming model -->
<mvc:annotation-driven />
<bean id="userManager" class="springapp.service.SimpleUserManager">
<property name="sessionFactory" ref="sessionFactory" />
</bean>
<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/" />
<property name="suffix" value=".jsp" />
</bean>
<import resource="hibernate-context.xml" />
</beans>
web.xml
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>/WEB-INF/springapp-servlet.xml</param-value>
</context-param>
<filter>
<filter-name>openSessionInView</filter-name>
<filter-class>org.springframework.orm.hibernate4.support.OpenSessionInViewFilter</filter-class>
<init-param>
<param-name>sessionFactoryBeanName</param-name>
<param-value>sessionFactory</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>openSessionInView</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<servlet>
<servlet-name>springapp</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>springapp</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
休眠上下文.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans ..
">
<context:property-placeholder location="/WEB-INF/hibernate.properties" />
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
<property name="driverClassName" value="${hibernate.connection.driver_class}" />
<property name="url" value="${hibernate.connection.url}" />
<property name="username" value="${hibernate.connection.username}" />
<property name="password" value="${hibernate.connection.password}" />
</bean>
<bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
<property name="sessionFactory" ref="sessionFactory"/>
</bean>
<tx:annotation-driven />
<bean id="transactionInterceptor" class="org.springframework.transaction.interceptor.TransactionInterceptor">
<property name="transactionManager" ref="transactionManager" />
<property name="transactionAttributeSource">
<bean class="org.springframework.transaction.annotation.AnnotationTransactionAttributeSource" />
</property>
</bean>
<bean id="managerTemplate" abstract="true" class="org.springframework.aop.framework.ProxyFactoryBean">
<property name="interceptorNames">
<list>
<value>transactionInterceptor</value>
</list>
</property>
</bean>
<bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
<property name="packagesToScan" value="databeans" />
<property name="dataSource" ref="dataSource" />
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">${hibernate.dialect}</prop>
</props>
</property>
</bean>
</beans>
管理器.java
@Service("userManager")
@Transactional
public class SimpleUserManager implements UserManager {
@Autowired
private SessionFactory sessionFactory;
public void setSessionFactory(SessionFactory sessionFactory){
this.sessionFactory = sessionFactory;
}
..
控制器.java
@Controller
@RequestMapping("/users")
public class UserController {
@Autowired
private UserManager userManager;
..
我的异常发生在控制器中,在读取管理器类中加载并传递给该控制器的对象的第一个属性时:
org.springframework.web.util.NestedServletException:请求处理失败;嵌套异常是 org.hibernate.LazyInitializationException: 无法初始化代理 - 没有会话 org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:894) org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java: 778)javax.servlet.http.HttpServlet.service(HttpServlet.java:617)javax.servlet.http.HttpServlet.service(HttpServlet.java:717) java:119) org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:76)
我已经研究了好几个小时了..我所能找到的只是使用休眠 3 的解决方案和对休眠 4 会话/事务理论的粗略解释,没有示例。
有什么想法吗?
更新:已修复!
发现这个:
Spring MVC OpenSessionInViewInterceptor 不工作
不幸的是,这并没有帮助那里的 OP,但它帮助了我。值得注意的是,这:
“最好不要在 dispatcher-servlet.xml 中导入 applicationContext.xml,而是使用 ContextLoaderListener 加载它”
其中,应用于我的配置,正是我在上面发布的所有内容,但不是
<import resource="hibernate-context.xml" />
在我的 springapp-servlet.xml 中,我将 ContextLoaderListener 修改为:
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>/WEB-INF/springapp-servlet.xml, /WEB-INF/hibernate-context.xml</param-value>
</context-param>
这就是全部
sessionFactory.getCurrentSession()
将 OSIV 创建的会话返回给我。而且,事实上,正在创建的第二个会话不再是,因为我发现了这些可爱的小日志行:
2012-10-04 14:43:48,743 DEBUG http-8080-1 org.springframework.beans.factory.support.DefaultListableBeanFactory - Returning cached instance of singleton bean 'transactionManager'
2012-10-04 14:43:48,743 TRACE http-8080-1 org.springframework.transaction.support.TransactionSynchronizationManager - Retrieved value [org.springframework.orm.hibernate4.SessionHolder@4146c5c0] for key [org.hibernate.internal.SessionFactoryImpl@12542011] bound to thread [http-8080-1]
2012-10-04 14:43:48,743 DEBUG http-8080-1 org.springframework.orm.hibernate4.HibernateTransactionManager - Found thread-bound Session [SessionImpl(PersistenceContext[entityKeys=[],collectionKeys=[]];ActionQueue[insertions=[] updates=[] deletions=[] collectionCreations=[] collectionRemovals=[] collectionUpdates=[] unresolvedInsertDependencies=UnresolvedEntityInsertActions[]])] for Hibernate transaction
2012-10-04 14:43:48,743 DEBUG http-8080-1 org.springframework.orm.hibernate4.HibernateTransactionManager - Creating new transaction with name [springapp.service.SimpleUserManager.find]: PROPAGATION_REQUIRED,ISOLATION_DEFAULT; ''
这表明,在我的 @Transactional 服务层之外,OSIV 创建的会话还活着并且运行良好,并且只是为它启动了一个新事务。
希望这对其他人有帮助。我什至不想考虑我花了多少时间在这上面工作。