对于一对多关联和多对多关联,应该优先考虑使用默认的延迟检索策略。在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集合代理类实例被初始化

在这里插入图片描述