《我的 O/R Mapping 之旅》勘误及补充

发表于:2007-06-22来源:作者:点击数: 标签:
《我的 O/R Mapping 之旅(二)》 ,有一段对 People.hbm.xml 的分析,谈到为什么使用 inverse=true: 在车辆管理系统中,代表着一个拥有者拥有多台车辆。以 java .util.Set 类型表示。 inverse 用于标识双向关联中的被动方一端。inverse=false 的一方(主控方

   

《我的 O/R Mapping 之旅(二)》,有一段对 People.hbm.xml 的分析,谈到为什么使用 inverse=”true”:

在车辆管理系统中,代表着一个拥有者拥有多台车辆。以 java.util.Set 类型表示。 inverse 用于标识双向关联中的被动方一端。inverse=false 的一方(主控方)负责维护关联关系;在车辆管理系统中, AutoInfo 作为主控方,应该把它设为“true”。这就好比你(被动方 one)在某个聚会上散发了许多名片,但是有可能你不清楚接收者(主动方 many)的具体背景;这个不要紧,接收者在必要的时候会和你联系就是了(主动维护关系)。

红色标识的句子容易让人产生歧异,好像是在说把 AtuoInfo 设置成“true”了。应改为:

在车辆管理系统中,AtuoInfo 作为主控方,应该在 People 中设置 inverse =“true”。

沿着思路往下走,你也许会问:什么才叫“主动维护关系”?不妨看看下面的代码(摘自《我的 O/R Mapping 之旅(三)》):

AutoInfo ai=new AutoInfo();
People people=new People();
public void DoTest() {
   try {
      Configuration cfg = new Configuration().configure();
      SessionFactory sessions = cfg.buildSessionFactory();
      Session session = sessions.openSession();
      Transaction tx = session.beginTransaction();
   
      ai.setLicensePlate("A00001");
      ai.setOwnerNo(people);
      people.setAddress("中国");
      people.setName("张三");
      people.addToAutoInfoSet(ai);
      session.save(people);
      tx.commit();
      session.close();
   } catch (Exception e) {
      System.out.println(e);
   }
}


把“ai.setOwnerNo(people)”注解了试试,由于 AutoInfo 没有主动维护关系,导致 AUTO_INFO 表中 OWNER_NO 字段为“Null”。自然 AutoInfo 与 Poople 就不存在任何联系了。

人类的求知欲很强烈!
为什么非要用 AutoInfo 作为主控方?People 作主控方不行?好吧,为 People.hbm.xml 删除inverse=”true”,再运行以上程序,其实也能保存,只是多了一条SQL:“update auto_info set OWNER_NO=? where AUTO_ID=?”,这就是 AutoInfo 被动地修改和 People 的联系。多执行一次 SQL 意味着多了一些开销,这是对性能不利的!


《我的 O/R Mapping 之旅(三)》,有一段对张三第二次买车的程序和描述:

 AutoInfo ai = new AutoInfo();
 People people = new People();
 public void DoTest() {
  try {
      Configuration cfg = new Configuration().configure();
      SessionFactory sessions = cfg.buildSessionFactory();
      Session session = sessions.openSession();
      Transaction tx = session.beginTransaction();
      people =
       (People) session
        .find(
         "from People where OWNER_ID=1")
        .get(0);
      ai.setLicensePlate("A00002");
      ai.setOwnerNo(people);
      people.getAutoInfoSet().add(ai);
      session.save(people);
      tx.commit();
      session.close();
  } catch (Exception e) {
      System.out.println(e);
  }
}


到这里,也许你会有这样的想法:“应该可以直接向 AUTO_INFO 表插入记录,不通过 People 对象中转,像写 SQL 一样 Easy。” 错了!以前直接写 SQL 是可以办到的,不过现在我们用的可是 Hibernate ,一切都要以对象行事,看见 ai.setOwnerNo(people) 了吗?传入参数是个 People 对象实例,不是简单的字段喔。

这段解释太绝对了,事实上可以直接保存 AutoInfo 对象,而不用通过保存 People 来中转:

AutoInfo ai = new AutoInfo();
People people = new People();
public void DoTest() {
  try {
      Configuration cfg = new Configuration().configure();
      SessionFactory sessions = cfg.buildSessionFactory();
      Session session = sessions.openSession();
      Transaction tx = session.beginTransaction();
      people =
       (People) session
        .find(
         "from People where OWNER_ID=1")
        .get(0);
      ai.setLicensePlate("A00002");
      ai.setOwnerNo(people);
      session.save(ai);
      tx.commit();
      session.close();
  } catch (Exception e) {
      System.out.println(e);
  }
}



《我的 O/R Mapping 之旅(四)》,删除 PEOPLE 表及其关联的 AUTO_INFO 表时,程序是没有错,不过有更简单的办法来删除:

  try {
   Configuration cfg = new Configuration().configure();
   SessionFactory sessions = cfg.buildSessionFactory();
   Session session = sessions.openSession();
   Transaction tx = session.beginTransaction();
   session.delete("from People where OWNER_ID=1");
   tx.commit();
   session.close();
  } catch (Exception e) {
   System.out.println(e);
  }

Hibernate 中,要完成一次操作,可以有多种实现方式,哪种最好,就要靠自己定夺了。


(请注意!引用、转贴本文应注明原作者:Rosen Jiang 以及出处:http://blog.csdn.net/rosen

原文转自:http://www.ltesting.net