用ehcache休眠不好用

时间:2015-04-28 23:01:36

标签: spring hibernate ehcache

我使用Spring 4,Hibernate 4作为我的项目。 最近,我在项目中添加了ehcache。有一个实体com.xxx.Employee,它拥有另一个实体com.xxx.User。我使用HibernateTemplate findByCriteria(DetachedCriteria,int,int)来执行查询,然后,通过HibernateTemplate get(entityName,id)进行另一个查询,在日志记录中没有显示SQL,这似乎是第二个缓存有效,但是,缺少Employee的用户,有" org.hibernate.LazyInitializationException:无法初始化代理 - 没有Session"抛出。

是否需要添加其他设置才能将对象的对象值保存到缓存中?

<hibernate-mapping package="com.xxx">
<class name="Employee" table="t_employee">
    <cache usage="read-write" include="all"/>
    <id name="personId" column="person_id">
        <generator class="identity" />
    </id>

    <property name="name" column="name" />

    <many-to-one name="department" column="department_id" fetch="join" not-null="false" />
    <many-to-one name="user" column="user_id" fetch="join" not-null="false" cascade="all-delete-orphan" unique="true"/>

</class>

<hibernate-mapping package="com.xxx">
<class name="User" table="t_user">
    <cache usage="read-write"/>
    <id name="userId" column="user_id">
        <generator class="identity" />
    </id>
    <property name="name" column="name" not-null="true" />
    <property name="loginName" column="login_name" unique="true" not-null="true" />
    <property name="password" column="password" not-null="true"  />
    <many-to-one name="employee" column="employee_id" fetch="join" not-null="false" />
    <many-to-one name="role" class="Role" column="role_id" fetch="join" not-null="true" />
    <property name="lock" column="col_lock" type="integer" />
    <property name="createDate" column="create_date" />
    <property name="updateDate" column="update_date" />
    <property name="passwordUpdateDate" column="pwd_update_date" />
    <property name="loginTimes" column="login_times" type="integer" />
    <property name="lastLoginDate" column="last_login_date" />
    <property name="comment" column="comment" />

</class>

public T getOneById(Serializable id) throws TamsException {
    if (id == null) {
        logger.error("getOneById id is null");
        throw new TamsException("getOneById id is null");
    }
    T t;
    try {
        log.info("get " + getEntityName() + " by ID : " + id);

        HibernateTemplate ht = getHibernateTemplate();
        ht.setCacheQueries(true);
        t = (T) ht.get(getEntityName(), id);
        return t;
    } catch (Exception e) {
        throw new TamsException("get " + getEntityName() + ":  failure!", e);
    } finally {
        log.info("get " + getEntityName() + " by ID : " + id + " success!");
    }
}


public List<T> queryByCondition(QueryCondition condition)
        throws TamsException {

    log.info("query " + getEntityName() + " by: " + condition + " ...");
    try {
        int firstResult = 0;
        int maxResults = 0;

        DetachedCriteria c = DetachedCriteria
                .forEntityName(getEntityName());

        if (condition != null) {
            firstResult = condition.getFirstResult();
            maxResults = condition.getMaxResults();
            assembleCriteria(condition, c);
        }
        HibernateTemplate ht = getHibernateTemplate();
        ht.setCacheQueries(true);
        @SuppressWarnings("unchecked")
        List<T> list = (List<T>) ht.findByCriteria(c, firstResult,
                maxResults);

        log.info("query " + getEntityName() + " by: " + condition
                + " ret size --> " + list.size());
        return list;
    } catch (Exception e) {
        throw new TamsException("query " + getEntityName() + " by: "
                + condition + " failure!", e);
    }

}

1 个答案:

答案 0 :(得分:0)

您收到org.hibernate.LazyInitializationException: could not initialize proxy - no Session错误,因为属性Employee.user配置了选项lazy=proxy(这是默认情况下,未提及时)。

以下是piece from documentation

  

lazy(可选 - 默认为代理):默认情况下,单点   协会代理。 lazy =“no-proxy”指定该属性   首次访问实例变量时应该懒惰地获取。   这需要构建时字节码检测。懒惰=“假”   指定始终紧急提取关联。

所以,现在你有两个选择:

  1. 使用lazy=false急切地获取关联
  2. 在使用必需的关联之前打开Session