主键相同的一对一关系
由于两个实体类是一对一的关系,因此可设置两个实体类使用相同的主键。反过来,具有相同主键的实体被视为一对一的关系。这样两个表就省掉了外键关联。
以客户地址管理为例学习主键相同的一对一关系:一个客户对应一个地址,对应的实体之间使用相同的主键。
客户与地址之间是一对一的关系,并且两个实体的id是相同的。双方都有对方的引用。注意,两个数据库表之间没有关联关系,hibernate是根据主键判断对应关系的。
Customer客户实体类中有一个id主键、一个name属性以及一个Address地址属性。其中Address地址属性与Customer客户实体类之间是一对一的属性,两者之间使用相同的主键,而没有使用传统的外键约束。用@OneToOne指定Address属性为一对一的,用@PrimaryKeyJoinColumn制定两个实体使用相同的主键。
Customer.java
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
Adderss地址实体类具有id主键、Customer客户属性、Adderss地址、zip邮编以及telephone电话号码等3个普通属性。Id主键使用@Id标注,因为实体间使用相同的id,因此不能使用任何的主键生成策略。Customer属性上用@OneToOne指定为一对一属性,并用@PrimaryKeyJoinColumn声明使用相同主键。
Address.java
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
测试方法:
不能使用数据库自动分配ID,自动分配的ID可能会是两个实体对象ID不一致。hibernate也不会自动保证两个实体ID相同,这时候要手工设置ID保证两个实体ID一致。测试代码如下:
TestCustomerAddress.java
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
注意:
程序中没有设置Address和Customer间的关系,但是查询Customer时仍能获取到对象的Address对象。因为hibernate会自动加载相同主键的另一边的对象。