(轉)Hibernate中inverse的用法

原文:http://blog.csdn.net/xiaoxian8023/article/details/15380529

一、Inverse是hibernate雙向關係中的基本概念。inverse的真正作用就是指定由哪一方來維護之間的關聯關係。當一方中指定了“inverse=false”(默認),那麼那一方就有責任負責之間的關聯關係,說白了就是hibernate如何生成Sql來維護關聯的記錄! 

       Hibernate僅僅按照主控方對象的狀態的變化來同步更新數據庫。按照原來的映射文件,people.getAddresses().add(address),即主控方對象的狀態發生了改變,因此數據庫會跟着對象狀態的變化來同步更新數據庫;而address.setPeople(people),即被控方對象的狀態發生了改變,它是不能觸發對象和數據庫的同步更新的。

      (實例1):

       舉個最簡單的一對多父子關係。那麼代碼就寫成:

 

Parent p = new Parent();
Child c = new Child();
c.setParent(p);  //維護父子之間關係
p.getChildren().add(c);

session.save(p);
session.flush();

 

 映射文件配置:

父親中的關係映射
{set name="children" lazy="true" inverse="true"}
      {key column="parent_id"/}
      {one-to-many class="test.Child"/}
{/set}

兒子中關係映射
{many-to-one name="parent" column="parent_id" not-null="true"/}

 

set中inverse="true",說明父子關係只在多的一端(Child)維護。所以只會發出2個insert語句。

       注意:{many-to-one}總是設成“inverse=false”的,而且這個屬性在Mapping中是不存在的!

       這樣運行的下來的結果就是:

 

Hibernate: insert into parent (id) values (?)
Hibernate: insert into child (parent_id, id) values (?, ?)

 

如果將set中的inverse設爲true,那麼會發出3條sql語句,前2條是insert語句,後1條是update語句用來維護parent和child類的父子關係。

       當然,假如c.setParent(p)註釋掉(破壞了父子關係),結果就變成了:

 

Hibernate: insert into parent (id) values (?)

 ===================================================
     (實例2):
       一個Person可以參加多個Event,一個Event有多個Person參加。映射文件如下:

 

<!-- Person.hbm.xml -->
<hibernate-mapping package="events">
  <class name="Person" table="person">
      <id name="id" column="person_id">
          <generator class="native"/>
      </id>
      <property name="age" length="0"/>
      <property name="firstname"/>
      <property name="lastname"/>
      <set name="events" table="person_event">
          <key column="person_id"/>
          <many-to-many column="event_id" class="events.Event"/>
      </set>
  </class>
</hibernate-mapping>

<!-- Event.hbm.xml -->
<hibernate-mapping>
  <class name="events.Event" table="events">
      <id name="id" column="event_id">
            <generator class="native"/>
      </id>
      <property name="date" column="events_date" type="timestamp"/>
      <property name="title" column="events_title"/>
      <set name="participants" table="person_event" inverse="true">
          <key column="event_id"/>
          <many-to-many column="person_id" class="events.Person"/>
      </set>
  </class>
</hibernate-mapping>

 inverse=true的含義: 由雙向關聯另一方維護該關聯,己方不維護該關聯(只能進行查詢操作)。在上述代碼中,由Person方維護該<many-to-many>關係,示例代碼如下(以向Person參與的Event中加入新的Event爲例):

 

 Session session = HibernateUtil.getSessionFactory().getCurrentSession();
        session.beginTransaction();
        Person p = (Person) session.load(Person.class, personId);
        Event e = (Event) session.load(Event.class, eventId);
        p.getEvents().add(e);//執行該代碼時,hibernate會向中間表 person_event中插入person_id和event_id記錄,如果換成e.getParticipants().add(p)的話,該代碼將不會被執行,即hibernate不會向表person_event中插入記錄。
        session.getTransaction().commit();

 要注意的一點:在雙向關聯的關係中,映射的column(和table)的值要一致(即要用相同的表名和列名),不然設置爲inverse="true"的這方將失去這個雙向關係,而變成了一個單向關聯。

       二、Inverse和Cascade的比較


       Inverse:負責控制關係,默認爲false,也就是關係的兩端都能控制,但這樣會造成一些問題,更新的時候會因爲兩端都控制關係,於是重複更新。一般來說有一端要設爲true。

 

       Cascade:負責控制關聯對象的級聯操作,包括更新、刪除等,也就是說對一個對象進行更新、刪除時,其它對象也受影響,比如我刪除一個對象,那麼跟它是多對一關係的對象也全部被刪除。
舉例說明區別:刪除“一”那一端一個對象O的時候,如果“多”的那一端的Inverse設爲true,則把“多”的那一端所有與O相關聯的對象外鍵清空;如果“多”的那一端的Cascade設爲Delete,則把“多”的那一端所有與O相關聯的對象全部刪除。

 

 

 

 

 

 

 

 

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章