您好我正在尝试实现一个简单的翻译列表,这意味着我有一个值和这个值的翻译。
[编辑:]因为这是我的用户界面的一部分,值和翻译可以通过xml导出,使用i18n文件接缝对这个命题非常不方便。这就是我决定将它们存储在数据库中的原因。
我有一个值类的域类:
class Value {
String label
static hasMany = [ translations: Translation ]
}
和一个具有唯一约束的翻译,以确保对于一个值,特定语言不得有多个翻译:
class Translation {
String value
Language language
static belongsTo = [ value: Value ]
static constraints = {
language(unique: 'value')
}
}
在为相同的值交换两种翻译语言后,我的问题出现了。 例如:
value.translations.each() { translation ->
println "${value.label} in ${translation.language.label} is ${translation.value}"
}
// process updates...
value.translations.each() { translation ->
println "${value.label} in ${translation.language.label} is ${translation.value}"
}
// validate...
打印出来
Comedy in german: Comedy
Comedy in english: Komödie
Comedy in english: Comedy
Comedy in german: Komödie
因此在更新之前和之后不会违反唯一约束,但无论如何我在保存时会遇到唯一约束失败。 另一个奇怪的事情是,当我在value上执行each()循环时,我只会收到此错误。如果我不检查内容,验证通过并且save(flush:true)方法返回true,但数据库中的值不会更改。
[编辑:]我认为当只有一个值被改变而另一个没有被改变时,问题出现在数据库级别上,因为恰好在该状态下违反了约束。如果更改将作为事务执行而不是在此中间步骤期间不检查约束,则可以避免这种情况。 (这可能是我想要的东西)
避免这种情况的另一种方法是删除并重新创建每个已编辑的bean,但我希望可能有更方便的方法来执行此操作。
感谢您的帮助
答案 0 :(得分:1)
当隐式或显式flush()
发生时,会检查约束。此时,GORM会检查数据库中是否存在另一个值。因此,如果一个实例已经flush()
编辑而另一个实例尚未编辑,则会出现约束违规。
在交易结束前尽量不要flush()
- 删除flush: true
参数,甚至将其设置为flush: false
。在交易结束时,这两项更改都应适用。
Grails中有一个警告,JFYTK:它在执行flush()
时执行隐式Criteria
,所以当你不打算{{1}时,不要对Hibernate错误感到太惊讶尚未。