【发布时间】:2017-07-30 11:22:53
【问题描述】:
我有这个实体用户,它有一个角色集合(集),我想对它进行 redis 缓存,所以每当它调用 getRoles() 时,它都会返回一个缓存副本。
目前的结构如下:
- User.roles 是 lazy="true" 和 access="field"。
- 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 时,即使没有实际变化。我正在寻找的是如何让休眠状态认为我的新集合就像检索到的会话(实际上不是)并且不认为它是脏的。
有什么解决方法或建议吗?
【问题讨论】:
-
@DraganBozanovic ummm.. 不,它与您链接的内容不同,因为它与加载子关系有关。
标签: java spring hibernate redis