java - 显式加载实体的集合

标签 java spring hibernate redis

我有这个实体 User,它有一个角色集合 (Set),我想对其进行缓存,因此每当它调用 getRoles() 时,它都会返回一个缓存的副本。

目前的结构如下:

  1. User.roles 是 lazy="true"和 access="field"。
  2. User.getRoles() 有一个调用 spring @Cacheable 方法的监听器 它从 Redis 缓存而不是二级缓存中检索。

映射信息:

<set
            name="roles"
            table="user_role"
            lazy="true"
            cascade="none"
            access="field"
        >
            <key
                column="user_id"
            >
            </key>
            <many-to-many
                class="com.resolution.scheduler.model.Role"
                column="role_id"
                outer-join="auto"
             />
        </set>

这是 GetRoles():

public Set getRoles() {
        if(!rolesUpdated  &&  this.id!=null){
            ApplicationContextProviderNonManageBean.getApplicationContext().publishEvent(
                 new com.resolution.scheduler.dao.event.UserRoleGetEvent(this));
        }
                 return roles;
    }

这是 UserRoleGetEvent 事件触发时调用的内容:

@Cacheable(value="userRoleByUserId" , key="#userId", condition="#userId!=null")
    public PersistentSet getUserRoleByUserId(final Long userId){
        if(userId==null){
            return new PersistentSet();
        }
        log.info("USer Role Getting from Database for User Id : "+userId);
        //final List<Role> roles=(List<Role>)getHibernateTemplate().find("Select u.roles from User u  where u.id=?",userId);
        final PersistentSet[] set= new PersistentSet[1];
        getHibernateTemplate().executeWithNativeSession(new HibernateCallback<Object>(){
            public Object doInHibernate(Session session) throws HibernateException{
                List roles=session.createQuery("Select u.roles from User u  where u.id=:userId").setParameter("userId",userId).list();
                set[0]= new PersistentSet((SessionImpl)session, new HashSet(roles));
                return null;
        }
        });
        log.info(set);

        return set[0];
    }

UserRoleGetEvent 的工作原理如下:

public void onApplicationEvent(UserRoleGetEvent event) {
        PersistentSet roles = userManager.getUserRoleByUserId(event.getUser().getId());
        event.getUser().setRoles((roles)); //**THIS_SET_MAKES_IT_DIRTY**

    }

问题: 当我执行 user.getRole() 时,因为它为 user.roles 设置了一个新集合并使 session 变脏。 在这个肮脏的 session 中,每当调用 flush 时,它都会删除所有角色并再次插入所有角色,即使没有实际更改。我正在寻找的是如何让 hibernate 状态认为我的新集合就像检索到的 session (实际上不是)并且不认为它是脏的。

有任何解决方法或建议吗?

最佳答案

Hibernate 不喜欢它的集合被改变。可能有用的是:

event.getUser().getRoles().clear();
event.getUser().getRoles().addAll(roles);

但是,我对您要尝试做的事情感到有点困惑。 User 上的 getRoles 会触发一个事件,然后更改角色?

作为旁注,我在这里是凭内存工作的,但您的代码可以重构为:

@Cacheable(value="userRoleByUserId" , key="#userId", condition="#userId!=null")
public List<Role> getUserRoleByUserId(final Long userId){
    if(userId==null){
        return Collections.emptyList();
    }
    log.info("USer Role Getting from Database for User Id : "+userId);

    @SuppressWarning("unchecked")
    List<Role> result = (List<Role>) getHibernateTemplate().executeWithNativeSession(new HibernateCallback<Object>(){
        public Object doInHibernate(Session session) throws HibernateException{
            return session.createQuery("Select u.roles from User u  where u.id=:userId").setParameter("userId",userId).list();
        }
    });
    log.info(set);

    return result;
}

public void onApplicationEvent(UserRoleGetEvent event) {
    List<Role> roles = userManager.getUserRoleByUserId(event.getUser().getId());
    event.getUser().getRoles().clear(); // yes, I know, getRoles will throw an event and send you in an infinite loop...  Fixing that just means having another method
    event.getUser().getRoles().addAll(roles);
}

关于java - 显式加载实体的集合,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42695954/

相关文章:

java - 无法停止android后台服务

java - Java 中的余弦相似性性能比等效的 C 慢 15 倍?

java - 嵌入式 Neo4j SPI 类 (lucene PostingsFormat) 错误

hibernate - 关于 Hibernate 缓存机制的一些说明?

java - Hibernate Session如果长时间不使用会释放db连接吗

java - 使用Java发送POST数据

java 属性文件对一个属性使用多行

java - Spring /hibernate : how to get Configuration from LocalContainerEntityManagerFactoryBean

java - 使用带有@Autowired 的Spring 托管bean 或带有静态方法的类?

java.lang.Object;无法转换到购物篮。产品