我正在使用JSF。最近我们决定从我们的项目中排除服务层,因为80%的情况下这个层只包括回忆相同的名为dao-methods(definelty,我知道,这种方式“不正确”)。所以,现在我们需要让我们的托管bean层事务化,这就是问题:我可以做dao-layer事务,但是当我尝试将这个职责移到managegbeans层时,我得到了一个
org.hibernate.HibernateException:找不到当前线程的会话
Dao和managedbeans图层是分开的模块,因此它们分离了 context.xml 文件。当我在dao的context.xml文件中定义 tx:annotation-driven 标记时,dao-module中的类中的 @Transactional 注释“visible” ,但是当我尝试在managedbeans context.xml文件中定义此标记时, @Transactional 注释(neo the dao,也不是managedbeans)都不可见,所以我得到一个例外。
任何想法,我如何正确配置交易?
的applicationContext-dao.xml
<jee:jndi-lookup id="dataSource" jndi-name="java:/MSSQLDataSource"/>
<bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
<property name="dataSource" ref="dataSource"/>
<property name="packagesToScan">
<list>
<value>com.company.project.entity</value>
</list>
</property>
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">org.hibernate.dialect.SQLServerDialect</prop>
<prop key="hibernate.show_sql">true</prop>
<prop key="hibernate.format_sql">true</prop>
</props>
</property>
</bean>
<bean id="txManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
<property name="sessionFactory" ref="sessionFactory"/>
</bean>
<context:component-scan base-package="com.company.project.dao"/>
的applicationContext-web.xml中
<context:annotation-config />
//Just because of the service layer is not fully removed, de facto we are using this import only for imort dao-context
<import resource="classpath:/applicationContext-service.xml"/>
<context:component-scan base-package="com.company.project"/>
<tx:annotation-driven transaction-manager="txManager" proxy-target-class="false"/>
<bean class="org.springframework.context.annotation.CommonAnnotationBeanPostProcessor" />
web.xml
的一部分 <context-param>
<param-name>contextConfigLocation</param-name>
<param-value>
classpath:/applicationContext-web.xml
classpath:/applicationContext-security.xml
</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<listener>
<listener-class>com.company.project.util.ServletContextListenerConfig</listener-class>
</listener>
<servlet>
<servlet-name>CXFServlet</servlet-name>
<servlet-class>
org.apache.cxf.transport.servlet.CXFServlet
</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet>
<servlet-name>Faces Servlet</servlet-name>
<servlet-class>javax.faces.webapp.FacesServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
ServletContextListenerConfig 类:
public class ServletContextListenerConfig implements ServletContextListener {
public void contextInitialized(ServletContextEvent event) {
System.setProperty("org.apache.el.parser.COERCE_TO_ZERO", "false");
}
public void contextDestroyed(ServletContextEvent event) {
// NOOP
}
}
提前致谢!
更新
在阅读了几篇文章后,我已经说过,这个问题可能出现在双定义bean中,因为它们用不同的<context: component-scan>
映射。在我读过的大多数文章中,这个上下文是contextConfigLocation
的根上下文和DispatcherServlet
的servlet上下文,但在我的情况下我没有DispatcherServlet
,但我有{ {1}} FacesServlet
。那么,Faces Servlet是否做同样的事情,如果是的话,怎么做?我没有在Faces Servlet和我的context.xml文件之间看到任何连接。
更新2: 我的错,我没有立刻显示这段代码。 这是我的控制器:
javax.faces.webapp.FacesServlet
@Component
@ViewScoped
public class ProfileBean extends EntityBeanBase<Profile> {
@Autowired
private ProfileDao profileDao;
@Override
@Transactional
protected List<Profile> getInitEntities() {
return profileDao.getAll();
}
public void profileNameValidator(FacesContext context, UIComponent component, Object value) {
if (!selectedEntity.getName().equals(value) && profileDao.exists((String) value)) {
System.out.println("Error validating");
}
}
由许多方法组成,但现在我只使用EntityBenBase
,这在getInitEntities()
中是抽象的。
和ProfileDaoImpl:
EnityBeanBase
BaseDao很简单:
@Repository
public class ProfileDaoImpl extends BaseDao implements ProfileDao {
@Override
public List<Profile> getAll() {
return sessionFactory.getCurrentSession().createCriteria(Profile.class).list();
}
..........
}