我有这个实体User并且它有一个我想要redis缓存的角色集合(Set),所以每当它调用getRoles()时,它都会返回一个缓存副本。
到目前为止,这是结构:
映射信息:
<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
所有roles
及insert
所有角色都会被重新调整。我正在寻找的是如何让hibernate认为我的新集合就像检索的会话(实际上并非如此)并且不认为它是脏的。
有任何变通方法或建议吗?
答案 0 :(得分:0)
Hibernate不喜欢它的集合被改变。可能有用的是:
event.getUser().getRoles().clear();
event.getUser().getRoles().addAll(roles);
然而,我对你想要做的事情感到有些困惑。 getRoles
上的User
会触发一个事件,然后更改角色?
作为旁注,我在这里通过内存工作,但你的代码可以重构为:
@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);
}