8.4. 组件作为联合标识符(Components as composite identifiers)

你可以使用一个组件作为一个实体类的标识符。 你的组件类必须满足以下要求:

  • 它必须实现java.io.Serializable接口

  • 它必须重新实现equals()hashCode()方法, 始终和组合关键字在数据库中的概念保持一致

注意:在Hibernate3中,第二个要求并非是Hibernate强制必须的。但最好这样做。

你不能使用一个IdentifierGenerator产生组合关键字。一个应用程序必须分配它自己的标识符。

使用<composite-id> 标签(并且内嵌<key-property>元素)代替通常的<id>标签。比如,OrderLine类具有一个主键,这个主键依赖于Order的(联合)主键。

  1. <class name="OrderLine">
  2. <composite-id name="id" class="OrderLineId">
  3. <key-property name="lineId"/>
  4. <key-property name="orderId"/>
  5. <key-property name="customerId"/>
  6. </composite-id>
  7. <property name="name"/>
  8. <many-to-one name="order" class="Order"
  9. insert="false" update="false">
  10. <column name="orderId"/>
  11. <column name="customerId"/>
  12. </many-to-one>
  13. ....
  14. </class>

现在,任何指向OrderLine的外键都是复合的。在你的映射文件中,必须为其他类也这样声明。例如,一个指向OrderLine的关联可能被这样映射:

  1. <many-to-one name="orderLine" class="OrderLine">
  2. <!-- the "class" attribute is optional, as usual -->
  3. <column name="lineId"/>
  4. <column name="orderId"/>
  5. <column name="customerId"/>
  6. </many-to-one>

(注意在各个地方&lt;column&gt;标签都是column属性的替代写法。)

指向OrderLine多对多关联也使用联合外键:

  1. <set name="undeliveredOrderLines">
  2. <key column name="warehouseId"/>
  3. <many-to-many class="OrderLine">
  4. <column name="lineId"/>
  5. <column name="orderId"/>
  6. <column name="customerId"/>
  7. </many-to-many>
  8. </set>

Order中,OrderLine的集合则是这样:

  1. <set name="orderLines" inverse="true">
  2. <key>
  3. <column name="orderId"/>
  4. <column name="customerId"/>
  5. </key>
  6. <one-to-many class="OrderLine"/>
  7. </set>

(与通常一样,&lt;one-to-many&gt;元素不声明任何列.)

假若OrderLine本身拥有一个集合,它也具有组合外键。

  1. <class name="OrderLine">
  2. ....
  3. ....
  4. <list name="deliveryAttempts">
  5. <key> <!-- a collection inherits the composite key type -->
  6. <column name="lineId"/>
  7. <column name="orderId"/>
  8. <column name="customerId"/>
  9. </key>
  10. <list-index column="attemptId" base="1"/>
  11. <composite-element class="DeliveryAttempt">
  12. ...
  13. </composite-element>
  14. </set>
  15. </class>