如何检查Grails中的瞬态字段的isDirty(' transient_fieldName')

时间:2015-06-22 05:55:39

标签: hibernate grails orm gorm

在我的一个域类中有一个案例,我们将一个瞬态字段存储到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
}

因此,最终的问题是我们如何调用瞬态字段修改的更新挂钩。

任何帮助都将受到高度赞赏。

1 个答案:

答案 0 :(得分:1)

实施Interceptor.findDirty

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语句中。