对于一对多关联和多对多关联,应该优先考虑使用默认的延迟检索策略。在Customer类中,以下代码对orders集合属性采用延迟检索策略:
//采用默认的延迟检索策略
@OneToMany(mappedBy="customer")
private Set<Order> orders = new HashSet<Order>();
或者:
//显式设置延迟检索策略
@OneToMany(mappedBy="customer",
fetch=FetchType.LAZY)
private Set<Order> orders = new HashSet<Order>();
此时运行EntityManager的find(Customer.class,Long.valueOf(1))方法,仅仅立即加载Customer对象,执行以下select语句:
select * from CUSTOMERS where ID=1;
EntityManager的find()方法返回的Customer对象的orders属性引用一个没有被初始化的集合代理类实例,换句话说,此时orders集合中没有存放任何Order对象。
只有当orders集合代理类实例被初始化时,才会到数据库中检索所有与Customer关联的Order对象,执行以下select语句:
select * from ORDERS where CUSTOMER_ID=1;
那么,Customer对象的orders属性引用的集合代理类实例什么时候被初始化呢?主要包括以下两种情况:
(1)当应用程序第一次访问它,例如调用它的iterator()、size()、isEmpty()或contains()方法:
Set<order> orders=customer.getOrders();
//导致orders集合代理类实例被初始化
Iterator<Order> it=orders.iterator();
(2)通过org.hibernate.Hibernate类的initialize()静态方法初始化它:
Set<Order> orders=customer.getOrders();
Hibernate.initialzie(orders); //导致orders集合代理类实例被初始化