GORM双向一对多

时间:2013-10-26 22:47:57

标签: grails gorm

我正在尝试在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;
   }
}

1 个答案:

答案 0 :(得分:2)

如果您read the documentation,您将了解到,正如您所发现但仍在质疑,将项目添加到集合以便正确保留的正确方法是使用addTo*方法

所以在你的情况下,当你说使用addToBars工作时,这是正确的方法。也就是说,你有一些性能打击。另一种方法是:

bar.foo = foo
bar.save()

缺点是foo在现有Set中不包含bar。你必须再次从数据库中提取它。这是一个给予和接受,你只是使用最好的方法适合你的情况。