如何使Spring托管bean事务性化

时间:2015-02-16 06:41:20

标签: spring hibernate jsf managed-bean

我正在使用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();
    }
..........
}

0 个答案:

没有答案