Візьмемо простий приклад. Скажімо, дві таблиці з назвою test
та customer
чи описані вони як:
create table test(
test_id int(11) not null auto_increment,
primary key(test_id));
create table customer(
customer_id int(11) not null auto_increment,
name varchar(50) not null,
primary key(customer_id));
Ще одна таблиця є там, яка веде облік test
s і customer
:
create table tests_purchased(
customer_id int(11) not null,
test_id int(11) not null,
created_date datetime not null,
primary key(customer_id, test_id));
Ми бачимо, що в таблиці tests_purchased
первинний ключ - це складений ключ, тому ми будемо використовувати <composite-id ...>...</composite-id>
тег у hbm.xml
файлі відображення. Так PurchasedTest.hbm.xml
буде виглядати:
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="entities.PurchasedTest" table="tests_purchased">
<composite-id name="purchasedTestId">
<key-property name="testId" column="TEST_ID" />
<key-property name="customerId" column="CUSTOMER_ID" />
</composite-id>
<property name="purchaseDate" type="timestamp">
<column name="created_date" />
</property>
</class>
</hibernate-mapping>
Але це не закінчується. У Hibernate ми використовуємо session.load ( entityClass
, id_type_object
) для пошуку та завантаження сутності за допомогою первинного ключа. У разі складених ключів об’єктом ID повинен бути окремий клас ID (у вище випадку PurchasedTestId
клас), який просто оголошує атрибути первинного ключа, як нижче :
import java.io.Serializable;
public class PurchasedTestId implements Serializable {
private Long testId;
private Long customerId;
// an easy initializing constructor
public PurchasedTestId(Long testId, Long customerId) {
this.testId = testId;
this.customerId = customerId;
}
public Long getTestId() {
return testId;
}
public void setTestId(Long testId) {
this.testId = testId;
}
public Long getCustomerId() {
return customerId;
}
public void setCustomerId(Long customerId) {
this.customerId = customerId;
}
@Override
public boolean equals(Object arg0) {
if(arg0 == null) return false;
if(!(arg0 instanceof PurchasedTestId)) return false;
PurchasedTestId arg1 = (PurchasedTestId) arg0;
return (this.testId.longValue() == arg1.getTestId().longValue()) &&
(this.customerId.longValue() == arg1.getCustomerId().longValue());
}
@Override
public int hashCode() {
int hsCode;
hsCode = testId.hashCode();
hsCode = 19 * hsCode+ customerId.hashCode();
return hsCode;
}
}
Важливим моментом є те, що ми також реалізуємо обидві функції, hashCode()
і equals()
коли сплячий режим покладається на них.