我正在尝试在GORM中创建Foo和Bar域之间的双向一对多关系。
这是我到目前为止所拥有的:
class Bar {
static belongsTo = [foo: Foo]
}
class Foo {
Set bars = []
static hasMany = [bars: Bar]
}
我遇到的问题是当我使用关系方法时,它们似乎没有像我认为的那样表现。例如,你会认为像“foo.bars.add(bar)”这样的语句也会在bar参数上设置foo字段。但是当我调用“bar.foo.id”时,我被告知foo字段为空。如果我使用“bar.foo = foo”而不是“foo.bars.add(bar)”,我可以解决这个问题。不幸的是,当我调用“foo.bars.size()”时,它告诉我它是0。
为了更清楚地了解我在说什么,这是我的测试:
def testFoo() {
def foo = new Foo()
def bar = new Bar()
foo.bars.add(bar)
println "foo.bars.size() = ${foo.bars.size()}"
println "bar.id = ${bar.id}"
for(def xbar : foo.bars) {
println "bar.id = ${xbar.id}"
}
println "foo.id = ${foo.id}"
println "bar.foo.id = ${bar?.foo?.id}" // <- is null
}
def testBar() {
def foo = new Foo()
def bar = new Bar()
bar.foo = foo
println "foo.bars.size() = ${foo.bars.size()}" // <- is 0
println "bar.id = ${bar.id}"
for(def xbar : foo.bars) {
println "bar.id = ${xbar.id}"
}
println "foo.id = ${foo.id}"
println "bar.foo.id = ${bar?.foo?.id}"
}
我做错了什么?
注意:我正在通过集成测试来运行它。我还发现“foo.addToBars(bar)”的工作方式与我认为“foo.bars.add(bar)”和“bar.foo = foo”应该有效。
更新以下是我做的快速黑客攻击(使用Hibernate和JPA):
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.LinkedList;
import stuff.Foo;
public class MyList<Bar> extends LinkedList {
private Foo foo;
public MyList(Foo foo) {
this.foo = foo;
}
@Override
public boolean add(Object obj) {
boolean result = super.add(obj);
try {
Method barMethod = obj.getClass().getDeclaredMethod("setFoo", Foo.class);
barMethod.invoke(obj, foo);
}
catch(NoSuchMethodException noSuchMethod) {
noSuchMethod.printStackTrace();
}
catch(InvocationTargetException invocationTarget) {
invocationTarget.printStackTrace();
}
catch(IllegalAccessException illegalAccess) {
illegalAccess.printStackTrace();
}
return result;
}
}
答案 0 :(得分:2)
如果您read the documentation,您将了解到,正如您所发现但仍在质疑,将项目添加到集合以便正确保留的正确方法是使用addTo*
方法
所以在你的情况下,当你说使用addToBars
工作时,这是正确的方法。也就是说,你有一些性能打击。另一种方法是:
bar.foo = foo
bar.save()
缺点是foo在现有Set中不包含bar。你必须再次从数据库中提取它。这是一个给予和接受,你只是使用最好的方法适合你的情况。