【问题标题】:Explicitly loading entity's collection显式加载实体的集合
【发布时间】:2017-07-30 11:22:53
【问题描述】:

我有这个实体用户,它有一个角色集合(集),我想对它进行 redis 缓存,所以每当它调用 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 设置了一个新集合,它使会话变脏。 在这个脏会话中,每当 flush 被称为 delete all roles 和它 insert all roles 时,即使没有实际变化。我正在寻找的是如何让休眠状态认为我的新集合就像检索到的会话(实际上不是)并且不认为它是脏的。

有什么解决方法或建议吗?

【问题讨论】:

标签: java spring hibernate redis


【解决方案1】:

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);
}

【讨论】:

  • getRoles() 有一个触发事件,调用 getUserRoleByUserId()。再次调用 getRoles() 将是 stackoverflow。
  • 是的。这就是为什么我说它会在行注释中。只需改用getRolesThatDontTriggerAnEvent 方法
猜你喜欢
  • 1970-01-01
  • 2013-07-11
  • 1970-01-01
  • 2011-10-20
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多