Spring MVC3とHibernate4.1を使用して宣言型トランザクション管理を取得し、リクエストの全期間にわたってオープンセッションを提供しようとしています。データアクセス用の「マネージャー」レイヤーは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>
hibernate-context.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>
manager.java
@Service("userManager")
@Transactional
public class SimpleUserManager implements UserManager {
@Autowired
private SessionFactory sessionFactory;
public void setSessionFactory(SessionFactory sessionFactory){
this.sessionFactory = sessionFactory;
}
..
controller.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)org.springframework.orm.hibernate4.support.OpenSessionInViewFilter.doFilterInternal(OpenSessionInViewFilter。 java:119)org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:76)
私はこれを何時間も研究してきました。私が見つけることができるのは、Hibernate3を使用したソリューションとHibernate4のセッション/トランザクション理論に関する詳細な説明を例なしで組み合わせたものだけです。
何かご意見は?
更新:修正済み!
これを見つけました:
SpringMVCOpenSessionInViewInterceptorが機能しない
残念ながら、そこでの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で作成されたセッションに戻ります。そして、実際、これらの素敵な小さなログ行を見つけたので、作成されていた2番目のセッションはもうありません。
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で作成されたセッションが正常に機能しており、そのために新しいトランザクションが開始されたことを示しています。
これが他の誰かに役立つことを願っています。何時間かけて取り組んだのか考えたくありません。