JPA 应用技巧 2:主键外键合体映射

 

JPA 应用技巧 2:主键外键合体映射

Posted on 2011-09-13 11:27 蜀山兆孨龘 阅读(1053) 评论(0)   编辑   收藏 所属分类: Java EE

考虑两个具有一对一关联的实体类,受控方除了有一个自己的主键,还有一个引用主控方的外键。如果主控方和受控方是同生同灭的关系,换句话说,双方的一对一关联一旦确立就不可更改,就可以考虑让双方共享相同的主键,简化受控方的表结构。下面就让楼主通过实例来说明如何用 JPA 2.0 来实现这种映射。

盯着眼前的电脑,楼主想到了一个也许不太贴切的例子:员工和公司配的电脑的关系。员工的主键就是工号。虽然电脑可能被换掉,但电脑实体完全可以用工号做主键,只是把电脑配置的详细信息改掉而已。此处的难点就在与如何将电脑的主键字段同时映射一个员工,请看楼主是怎么一步步推导出来的。

一开始是最想当然的写法:

?
1
2
3
4
5
6
public class Computer implements Serializable {
     @Id
     @OneToOne
     private Employee employee;
     // 此处省略若干行
}

然而根据规范,只有这些类型可以作为主键:Java 原始类型(例如 int)、原始包装类型(例如 Integer)、Stringjava.util.Datejava.sql.Datejava.math.BigDecimaljava.math.BigInteger。所以直接拿 Employee 做主键是不行的。顺便提一下,也许某些 JPA 实现自己做了扩展,使得可以直接拿实体类做主键,这已经超出了 JPA 规范的范畴,此处不讨论。

直接映射是行不通了,那有什么间接的方式吗?这时楼主想起了一个特殊的注解:EmbeddedId。该注解的本意是用于联合主键,不过变通一下,是否可以将Employee 包装进一个嵌入式主键,然后再将这个嵌入式主键作为 Computer 的主键以达到目的?带着这种想法,楼主有了下面的代码:

?
1
2
3
4
5
6
@Embeddable
public class ComputerId implements Serializable {
     @OneToOne
     private Employee employee;
     // 此处省略若干行
}
?
1
2
3
4
5
public class Computer implements Serializable {
     @EmbeddedId
     private ComputerId id;
     // 此处省略若干行
}

现在又出现了新的问题:JPA 不支持定义在嵌入式主键类中的关联映射。好在天无绝人之路,EmbeddedId 的文档中直接指出,可以使用MapsId 注解来间接指定嵌入式主键类中的关联映射,而且还附带了一个例子!于是最终的成品就出炉了:

?
1
2
3
4
5
@Embeddable
public class ComputerId implements Serializable {
     private String employeeId;
     // 此处省略若干行
}
?
1
2
3
4
5
6
7
8
public class Computer implements Serializable {
     @EmbeddedId
     private Employee employee;
     @MapsId ( "employeeId" )
     @OneToOne
     private Employee employee;
     // 此处省略若干行
}

唯一的遗憾是,逻辑上的主控方 Employee 现在不得不成为受控方了,好在可以定义级联操作来达到差不多的效果:

?
1
2
3
4
5
6
7
public class Employee implements Serializable {
     @Id
     private String id;
     @OneToOne (mappedBy = "employee" , cascade = CascadeType.ALL)
     private Computer computer;
     // 此处省略若干行
}

虽然做到了,但确实挺绕的。希望未来版本的 JPA 能直接支持将实体类作为主键,楼主个人觉得不是一个技术问题。

你可能感兴趣的:(JPA 应用技巧 2:主键外键合体映射)