1.3.3. 使关联工作

我们把一些people和events 一起放到EventManager的新方法中:

  1. private void addPersonToEvent(Long personId, Long eventId) {
  2. Session session = HibernateUtil.getSessionFactory().getCurrentSession();
  3. session.beginTransaction();
  4. Person aPerson = (Person) session.load(Person.class, personId);
  5. Event anEvent = (Event) session.load(Event.class, eventId);
  6. aPerson.getEvents().add(anEvent);
  7. session.getTransaction().commit();
  8. }

在加载一PersonEvent后,使用普通的集合方法就可容易地修改我们定义的集合。如你所见,没有显式的update()save(),Hibernate会自动检测到集合已经被修改并需要更新回数据库。这叫做自动脏检查(automatic dirty checking),你也可以尝试修改任何对象的name或者date属性,只要他们处于持久化状态,也就是被绑定到某个Hibernate 的Session上(如:他们刚刚在一个单元操作被加载或者保存),Hibernate监视任何改变并在后台隐式写的方式执行SQL。同步内存状态和数据库的过程,通常只在单元操作结束的时候发生,称此过程为清理缓存(flushing)。在我们的代码中,工作单元由数据库事务的提交(或者回滚)来结束——这是由CurrentSessionContext类的thread配置选项定义的。

当然,你也可以在不同的单元操作里面加载person和event。或在Session以外修改不是处在持久化(persistent)状态下的对象(如果该对象以前曾经被持久化,那么我们称这个状态为脱管(detached))。你甚至可以在一个集合被脱管时修改它:

  1. private void addPersonToEvent(Long personId, Long eventId) {
  2. Session session = HibernateUtil.getSessionFactory().getCurrentSession();
  3. session.beginTransaction();
  4. Person aPerson = (Person) session
  5. .createQuery("select p from Person p left join fetch p.events where p.id = :pid")
  6. .setParameter("pid", personId)
  7. .uniqueResult(); // Eager fetch the collection so we can use it detached
  8. Event anEvent = (Event) session.load(Event.class, eventId);
  9. session.getTransaction().commit();
  10. // End of first unit of work
  11. aPerson.getEvents().add(anEvent); // aPerson (and its collection) is detached
  12. // Begin second unit of work
  13. Session session2 = HibernateUtil.getSessionFactory().getCurrentSession();
  14. session2.beginTransaction();
  15. session2.update(aPerson); // Reattachment of aPerson
  16. session2.getTransaction().commit();
  17. }

update的调用使一个脱管对象重新持久化,你可以说它被绑定到一个新的单元操作上,所以在脱管状态下对它所做的任何修改都会被保存到数据库里。这也包括你对这个实体对象的集合所作的任何改动(增加/删除)。

这对我们当前的情形不是很有用,但它是非常重要的概念,你可以把它融入到你自己的应用程序设计中。在EventManager的main方法中添加一个新的动作,并从命令行运行它来完成我们所做的练习。如果你需要person及event的标识符 — 那就用save()方法返回它(你可能需要修改前面的一些方法来返回那个标识符):

  1. else if (args[0].equals("addpersontoevent")) {
  2. Long eventId = mgr.createAndStoreEvent("My Event", new Date());
  3. Long personId = mgr.createAndStorePerson("Foo", "Bar");
  4. mgr.addPersonToEvent(personId, eventId);
  5. System.out.println("Added person " + personId + " to event " + eventId);
  6. }

上面是个关于两个同等重要的实体类间关联的例子。像前面所提到的那样,在特定的模型中也存在其它的类和类型,这些类和类型通常是“次要的”。你已看到过其中的一些,像intString。我们称这些类为值类型(value type),它们的实例依赖(depend)在某个特定的实体上。这些类型的实例没有它们自己的标识(identity),也不能在实体间被共享(比如,两个person不能引用同一个firstname对象,即使他们有相同的first name)。当然,值类型并不仅仅在JDK中存在(事实上,在一个Hibernate应用程序中,所有的JDK类都被视为值类型),而且你也可以编写你自己的依赖类,例如AddressMonetaryAmount

你也可以设计一个值类型的集合,这在概念上与引用其它实体的集合有很大的不同,但是在Java里面看起来几乎是一样的。