重点牢记
1.Load支持懒加载,get不支持懒加载。
2.lazy的生命周期与session相同,lazy加载必须依赖于session一直开启。
3. lazy属性,在3.x后是默认打开的,在以前版本中默认是关闭的。
4.hibernate通过cjlib实现代理。
5.load方法加载出来的是代理对象。
6.可以利用Hibernate.initialize(emp)来初始化代理对象达到命中
7.代理类在未命中数据库之前,他只有真实对象的ID属性,其他数据都是没有的
Empoyee.hbm.xml的配置
Department.hbm.xml的配置
(一):
public class Many2One { /** * @param args */ public static void main(String[] args) { add(); Employee employee =query(); System.out.println(employee.getDepartment().getName());//(1) 进行访问 } /** * 添加一个Department 和Employee 以及它们之间的关联 */ static void add(){ Session s =null; Transaction t = null; s=HiberUtil.getSession(); t=s.beginTransaction(); Department department = new Department(); department.setName("人事部"); Employee employee = new Employee(); employee.setName("steve"); employee.setDepartment(department); s.save(department); s.save(employee); t.commit(); s.close(); } /** * 使用load方法测试 懒加载的相关问题 * @return */ static Employee query(){ Session s =null; Transaction t = null; s=HiberUtil.getSession(); t=s.beginTransaction(); Employee employee = (Employee)s.load(Employee.class, 1); t.commit(); s.close(); return employee; } }
结果: 会报错
Hibernate: select max(id) from departments Hibernate: select max(id) from employees Hibernate: insert into departments (name, id) values (?, ?) Hibernate: insert into employees (name, department_id, id) values (?, ?, ?) Exception in thread "main" org.hibernate.LazyInitializationException: could not initialize proxy - no Session at org.hibernate.proxy.AbstractLazyInitializer.initialize(AbstractLazyInitializer.Java:132) at org.hibernate.proxy.AbstractLazyInitializer.getImplementation(AbstractLazyInitializer.java:174) at org.hibernate.proxy.pojo.javassist.JavassistLazyInitializer.invoke(JavassistLazyInitializer.java:190) at hiber.domain.Employee__javassist_0.getDepartment(Employee__javassist_0.java) at hiber.sth.Many2One.main(Many2One.java:17)
结果分析:
当使用session中的load方法查询数据库中的记录时,我们返回的是一个代理对象,而不是真正需要的那个对象;例如数据库中存有个Employee表,我们有cn.binyulan.doman.Employee的领域对象,如果查询Id值为“200626313”的Employee employee = (Employee)session.load(Employee.class,"200626313"),然后我们打印System.out.println(employee.getClass);得到的结果为cn.binyulan.domain.Employee._$$_jvst793_0,名字很奇怪吧,这个对象是Hibernate帮我们生成的,从名字可以看出它是对Employee类的增强类的对象,其实这就是个代理对象,这个对象里并没有我们需要的Employee的数据,所以如果你在session关闭后在使用employee来获取信息,如除了获得ID以外的employee.getName();就会出现如下异常:org.hibernate.LazyInitializationException: could not initialize proxy - no Session,这就说明了employee中没有我们需要的数据了。
emp对象现在到底是什么呢,它其实是一个代理类,这个类具有查询数据库的能力,当session没有关闭的时候如果我们调用emp.getName等方法;那么这个类会去查询数据库并返回相应的数据。之后我们关闭session之后再去使用这个类就不会有异常了,可是我们如果调用emp.getName()只是为了让代理类去查数据库,如果别人看我们的代码时候觉得这两句在逻辑上根本就没有用,别人就很容易注释掉,一注释掉就又会出现异常了,所以hibernate提供了一个方法Hibernate.initialize(emp);这样就可以初始化这个代理对象了。
不知道大家注意到没有,上面的那句代码Employee emp = (Employee)session.load(Employee.class,1)把emp强制转化为Employee类型,有人会问:不是返回的是代理对象么,那怎么又强制转换成Employee了呢,其实代理类是Employee的子类,它具有了比父类更强的能力(数据库查询),这个类是怎么生成的呢?其实hibernate使用了asm.jar和cglig-2.1.3.jar,在内存中修改Employee类的字节码,修改后的字节码只要符合class文件的规则,就可以创建出代理对象。
Domain对象不应该final的,大家现在应该明白为什么了吧,如果是final的,那么就不可以继承,当然也就不可以产生代理对象,也就不能实现懒加载了,如果你不用懒加载,那么把domain对象设计成fianl的也是可以的。