在我的一个域类中有一个案例,我们将一个瞬态字段存储到beforeInsert和afterUpdate钩子中的Mongo数据库,它完全符合以下条件: -
问题是isDirty适用于非瞬态属性。
代码如下:
class ResoruceInstance {
def configurationService
Status status
//Transient
Map<String, Object> configuration
static transients = ['configuration']
public Map<String, Object> getConfiguration() {
if(!configuration)
configuration = configurationService.get(id, CollectionConstants.RESOURCE_INSTANCE_IDENTIFIER)
return configuration
}
def afterInsert() {
configurationService.save(id, CollectionConstants.RESOURCE_INSTANCE_IDENTIFIER, configuration)
}
def afterUpdate() {
if(this.isDirty("configuration"))
configurationService.save(id, CollectionConstants.RESOURCE_INSTANCE_IDENTIFIER, configuration)
}
}
为了解决这个问题,我创建了isDirtyMongo(&#39; transient_field&#39;)。这非常有效,直到修改非瞬态属性为止,因为仅为瞬态属性调用afterUpdate。
修改后的钩子如下:
def afterUpdate() {
if(this.isDirtyMongo("configuration"))
configurationService.save(id, CollectionConstants.RESOURCE_INSTANCE_IDENTIFIER, configuration)
}
boolean isDirtyMongo(String property){
//return whether is dirty or not
}
因此,最终的问题是我们如何调用瞬态字段修改的更新挂钩。
任何帮助都将受到高度赞赏。
答案 0 :(得分:1)
public class TransientFieldDirtinessInterceptor extends EmptyInterceptor {
@Override
public int[] findDirty(Object entity, ..., String[] propertyNames, ...) {
if ((entity instanceof EntityToCheck) && isTransientFieldDirty(entity)) {
// Just return all fields as dirty
int[] result = new int[propertyNames.length];
for(int i = 0; i < result.length; i++) {
result[i] = i;
}
return result;
}
// Use Hibernate's default dirty-checking algorithm
return null;
}
}
基本上,如果瞬态字段是脏的,让Hibernate认为所有字段都是脏的。
您可以尝试稍微优化一下,将第一个属性标记为脏(无论其中有多少是脏的,如果至少有一个属性是脏的,则实体是脏的):
int[] result = new int[1];
result[0] = 0;
return result;
但是,如果对这些实体使用@DynamicUpdate
,这将始终排除SQL update
语句中的其他属性,因此我认为更清晰一致的方法是将所有属性标记为脏(没有@DynamicUpdate
所有属性总是包含在SQL update
语句中。