:雙向一對(duì)多關(guān)系,一是關(guān)系維護(hù)端(owner side),多是關(guān)系被維護(hù)端(inverse side)。在關(guān)系被維護(hù)端需要通過(guò)@JoinColumn建立外鍵列指向關(guān)系維護(hù)端的主鍵列。
publicclass Order implements Serializable {
privateSet<OrderItem> orderItems = new HashSet<OrderItem>();
。。。。
@OneToMany(mappedBy="order",cascade = CascadeType.ALL, fetch = FetchType.LAZY)
@OrderBy(value = "id ASC")
public Set<OrderItem> getOrderItems() {
returnorderItems;
}
}
publicclass OrderItem implements Serializable {
private Order order;
。。。。
@ManyToOne(cascade=CascadeType.REFRESH,optional=false)
@JoinColumn(name = "order_id")
public Order getOrder() {
returnorder;
}
}
@OrderBy(value = "id ASC") 指明加載OrderItem 時(shí)按id 的升序排序
@OneToMany的屬性:
1>targetEntity
定義關(guān)系類的類型,默認(rèn)是該成員屬性對(duì)應(yīng)的類類型,所以通常不需要提供定義。
2>mappedBy
定義類之間的雙向關(guān)系。如果類之間是單向關(guān)系,不需要提供定義,如果類和類之間形成雙向關(guān)系,我們就需要使用這個(gè)屬性進(jìn)行定義,否則可能引起數(shù)據(jù)一致性的問(wèn)題。
該屬性的值是“多”方class里的“一”方的變量名
3>cascade
該屬性定義類和類之間的級(jí)聯(lián)關(guān)系。定義的級(jí)聯(lián)關(guān)系將被容器視為對(duì)當(dāng)前類對(duì)象及其關(guān)聯(lián)類對(duì)象采取相同的操作,而且這種關(guān)系是遞歸調(diào)用的。舉個(gè)例子:Order 和OrderItem有級(jí)聯(lián)關(guān)系,那么刪除Order時(shí)將同時(shí)刪除它所對(duì)應(yīng)的OrderItem對(duì)象。而如果OrderItem還和其他的對(duì)象之間有級(jí)聯(lián)關(guān)系,那么這樣的操作會(huì)一直遞歸執(zhí)行下去。
cascade的值只能從CascadeType.PERSIST(級(jí)聯(lián)新建)、CascadeType.REMOVE(級(jí)聯(lián)刪除)、CascadeType.REFRESH(級(jí)聯(lián)刷新)、CascadeType.MERGE(級(jí)聯(lián)更新)中選擇一個(gè)或多個(gè)。還有一個(gè)選擇是使用CascadeType.ALL,表示選擇全部四項(xiàng)。
4>fatch
可選擇項(xiàng)包括:FetchType.EAGER和FetchType.LAZY。前者表示關(guān)系類(本例是OrderItem 類)在主類(本例是Order類)加載的時(shí)候同時(shí)加載,后者表示關(guān)系類在被訪問(wèn)時(shí)才加載。默認(rèn)值是FetchType.LAZY。
@JoinColumn(name = "order_id")注釋指定OrderItem映射表的order_id列作為外鍵與Order 映射表的主鍵列關(guān)聯(lián)。
@ManyToOne:指明OrderItem和Order之間為多對(duì)一關(guān)系。
@ManyToOne注釋有四個(gè)屬性:targetEntity、cascade、fetch 和optional,前三個(gè)屬性的具體含義和@OneToMany的同名屬性相同,但@ManyToOne的fetch 屬性默認(rèn)值是FetchType.EAGER。
optional屬性是定義該關(guān)聯(lián)類是否必須存在,值為false 時(shí),關(guān)聯(lián)類雙方都必須存在,如果關(guān)系被維護(hù)端不存在,查詢的結(jié)果為null。值為true 時(shí), 關(guān)系被維護(hù)端可以不存在,查詢的結(jié)果仍然會(huì)返回關(guān)系維護(hù)端,在關(guān)系維護(hù)端中指向關(guān)系被維護(hù)端的屬性為null。optional屬性的默認(rèn)值是true。optional 屬性實(shí)際上指定關(guān)聯(lián)類與被關(guān)聯(lián)類的join 查詢關(guān)系,如optional=false 時(shí)join 查詢關(guān)系為inner join, optional=true 時(shí)join 查詢關(guān)系為left join。下面代碼片斷解釋如下:
有一點(diǎn)需要強(qiáng)調(diào):當(dāng)業(yè)務(wù)方法需要把一個(gè)實(shí)體Bean作為參數(shù)返回給客戶端時(shí),除了實(shí)體Bean本身需要實(shí)現(xiàn)Serializable 接口之外,如果關(guān)聯(lián)類(OrderItem)是延遲加載,還需在返回實(shí)體Bean之前通過(guò)訪問(wèn)關(guān)聯(lián)類的方式加載關(guān)聯(lián)類(見(jiàn)下例)。否則在客戶端訪問(wèn)關(guān)聯(lián)類時(shí)將會(huì)拋出加載例外。
public Order getOrderByID(Integer orderid) {
Order order = em.find(Order.class, orderid);
//!!!!!因?yàn)槭茄舆t加載,通過(guò)執(zhí)行size()這種方式獲取訂單下的所有訂單項(xiàng)
order.getOrderItems().size();
return order;
}
另外不管是否延遲加載,通過(guò)join fetch 關(guān)聯(lián)語(yǔ)句都可顯式加載關(guān)聯(lián)類,如下例:
public List getAllOrder() {
Query query = em.createQuery("select DISTINCT o from Order o inner
join fetch o.orderItems order by o.orderid");
List result = query.getResultList();
return result;
}