spring-hibernate多用户web-app中的DAO范围?

时间:2011-03-15 13:50:02

标签: hibernate spring

我实际上看到了this question,但是从中得不到多少,所以我会尝试更加具体地了解它。
我的多用户网络应用程序中有 BaseDAO 类,如下所示:

public abstract class BaseDAO<GenType>
{
private HibernateOperations hibernateTemplate;

protected BaseDAO() {}
protected HibernateOperations getHibernateTemplate() {
    return hibernateTemplate;
}
public void setHibernateTemplate(HibernateOperations hibernateTemplate) {
    this.hibernateTemplate = hibernateTemplate;
}
protected void initialize(final Object proxy) throws DataAccessException {
    hibernateTemplate.initialize(proxy);
}
public GenType merge(GenType entity) throws DataAccessException {
    return (GenType)hibernateTemplate.merge(entity);
}
protected void persist(GenType entity) throws DataAccessException {
    hibernateTemplate.persist(entity);
}
public void refresh(GenType entity) throws DataAccessException {
    hibernateTemplate.refresh(entity);
}
public void save(GenType entity) throws DataAccessException {
    hibernateTemplate.save(entity);
}
public void saveOrUpdate(GenType entity) throws DataAccessException {
    hibernateTemplate.saveOrUpdate(entity);
}
public void update(GenType entity) throws DataAccessException {
    hibernateTemplate.update(entity);
}
public void delete(GenType entity) throws DataAccessException {
    hibernateTemplate.delete(entity);
}
protected void deleteAll(Collection<GenType> entities) throws DataAccessException {
    hibernateTemplate.deleteAll(entities);
}
protected GenType get(Class<GenType> entityClass, Serializable id) throws DataAccessException {
    return (GenType)hibernateTemplate.get(entityClass, id);
}
}

它基本上是围绕 HibernateTemplate 的包装。我所有的其他DAO都继承了这个类并实现了适当的接口,这些接口包含一些额外的方法(比如getBySomeAttribute())。所以基本上这些DAO只有 方法。此外,我有服务层包装DAO。也就是说,服务类可以容纳多个DAO,并且使用spring-AOP拦截来自服务层的方法调用以进行自动提交/回滚(事务去除)。例如:

public class ModelDAO extends BaseDAO<Model> implements IModelDAO
{
    @Override
    public Model getNew() {
        return new Model();
    }

    @Override
    public List<Model> getBySomeAttr() {
        DetachedCriteria criteria;
        // define some criteria
        return getHibernateTemplate().findByCriteria(criteria);
    }
}

...

public class ModelService
{
    private ModelDAO modelDAO;
    private ElementDAO elementDAO;
    // GET/SET for model and user DAO

    public void doSomethingWithModel() {

        modelDAO.doSomething();
        elementDAO.doSomethingElse();
    }
}

配置如下:

<bean id="hibernateTemplate2" class="org.springframework.orm.hibernate3.HibernateTemplate">
    <property name="sessionFactory" ref="sessionFactory2" />
    <property name="maxResults" value="3000" />
</bean>

<!-- DAO -->
<bean id="baseDAO" abstract="true" 
    class="org.irvas.backend.dao_implement.BaseDAO" scope="session">
    <property name="hibernateTemplate" ref="hibernateTemplate2" />
</bean>
<bean id="modelDAO" 
    class="org.irvas.backend.dao_implement.ModelDAOImplement" 
    parent="baseDAO" scope="session">
</bean>
<bean id="elementDAO" 
    class="org.irvas.backend.dao_implement.ElementDAOImplement" 
    parent="baseDAO" scope="session">
</bean>

<!-- Service -->
<bean id="modelService" 
    class="org.irvas.backend.service_implement.ModelServiceImplement" scope="session">
    <property name="modelDAO" ref="modelDAO" />
    <property name="elementDAO" ref="elementDAO" />
</bean>
<bean id="elementService" 
    class="org.irvas.backend.service_implement.ElementServiceImplement" scope="session">
    <property name="elementDAO" ref="elementDAO" />
</bean>

所以,我想知道如何为多用户目的设置DAO / Service bean的范围?我正在将服务bean注入GUI的原型控制器。 当我使用这个配置(使用scope =“session”)时,我得到如下错误:

  

创建名为'modelService'的bean时出错:java.lang.IllegalArgumentException setAttribute:非可序列化属性:modelDAO

我会说这是从Tomcat的StandardSession.setAttribute()抛出的。从这一点我可以得出结论,我的DAO应该实现Serializable,更令我困惑的是,我看到完全相同的代码无法实现Serializable ...

如果有人能够告诉我这里发生了什么,并且如何为这个特定案例规划DAO和服务的范围,我将非常感激......

1 个答案:

答案 0 :(得分:3)

没有必要让你的DAO会话作用域。

虽然Hibernate会话具有状态,但在正确配置的事务管理下,其状态绑定到事务(即,在相同事务内部调用的不同DAO方法共享同一会话,而从不同事务调用的相同方法使用不同的会话)。 / p>

因此,您的DAO实际上是无状态的,应该是单例范围的(即默认范围)。